ClickHouse/docs/zh/sql-reference/functions/date-time-functions.md
2021-01-18 12:16:32 +08:00

27 KiB
Raw Blame History

时间日期函数

支持时区。

所有的时间日期函数都可以在第二个可选参数中接受时区参数。示例Asia / Yekaterinburg。在这种情况下它们使用指定的时区而不是本地默认时区。

SELECT
    toDateTime('2016-06-15 23:00:00') AS time,
    toDate(time) AS date_local,
    toDate(time, 'Asia/Yekaterinburg') AS date_yekat,
    toString(time, 'US/Samoa') AS time_samoa
┌────────────────time─┬─date_local─┬─date_yekat─┬─time_samoa──────────┐
│ 2016-06-15 23:00:00 │ 2016-06-15 │ 2016-06-16 │ 2016-06-15 09:00:00 │
└─────────────────────┴────────────┴────────────┴─────────────────────┘

仅支持与UTC相差一整小时的时区。

toTimeZone

将Date或DateTime转换为指定的时区。 时区是Date/DateTime类型的属性。 表字段或结果集的列的内部值(秒数)不会更改,列的类型会更改,并且其字符串表示形式也会相应更改。

SELECT
    toDateTime('2019-01-01 00:00:00', 'UTC') AS time_utc,
    toTypeName(time_utc) AS type_utc,
    toInt32(time_utc) AS int32utc,
    toTimeZone(time_utc, 'Asia/Yekaterinburg') AS time_yekat,
    toTypeName(time_yekat) AS type_yekat,
    toInt32(time_yekat) AS int32yekat,
    toTimeZone(time_utc, 'US/Samoa') AS time_samoa,
    toTypeName(time_samoa) AS type_samoa,
    toInt32(time_samoa) AS int32samoa
FORMAT Vertical;
Row 1:
──────
time_utc:   2019-01-01 00:00:00
type_utc:   DateTime('UTC')
int32utc:   1546300800
time_yekat: 2019-01-01 05:00:00
type_yekat: DateTime('Asia/Yekaterinburg')
int32yekat: 1546300800
time_samoa: 2018-12-31 13:00:00
type_samoa: DateTime('US/Samoa')
int32samoa: 1546300800

toTimeZone(time_utc, 'Asia/Yekaterinburg')DateTime('UTC') 类型转换为 DateTime('Asia/Yekaterinburg'). 内部值 (Unixtimestamp) 1546300800 保持不变, 但是字符串表示(toString() 函数的结果值) 由 time_utc: 2019-01-01 00:00:00 转换为o time_yekat: 2019-01-01 05:00:00.

toYear

将Date或DateTime转换为包含年份编号AD的UInt16类型的数字。

toQuarter

将Date或DateTime转换为包含季度编号的UInt8类型的数字。

toMonth

将Date或DateTime转换为包含月份编号1-12的UInt8类型的数字。

toDayOfYear

将Date或DateTime转换为包含一年中的某一天的编号的UInt161-366类型的数字。

toDayOfMonth

将Date或DateTime转换为包含一月中的某一天的编号的UInt81-31类型的数字。

toDayOfWeek

将Date或DateTime转换为包含一周中的某一天的编号的UInt8周一是1, 周日是7类型的数字。

toHour

将DateTime转换为包含24小时制0-23小时数的UInt8数字。 这个函数假设如果时钟向前移动它是一个小时发生在凌晨2点如果时钟被移回它是一个小时发生在凌晨3点这并非总是如此 - 即使在莫斯科时钟在不同的时间两次改变)。

toMinute

将DateTime转换为包含一小时中分钟数0-59的UInt8数字。

toSecond

将DateTime转换为包含一分钟中秒数0-59的UInt8数字。 闰秒不计算在内。

toUnixTimestamp

对于DateTime参数将值转换为UInt32类型的数字-Unix时间戳https://en.wikipedia.org/wiki/Unix_time。 对于String参数根据时区将输入字符串转换为日期时间可选的第二个参数默认使用服务器时区并返回相应的unix时间戳。

语法

toUnixTimestamp(datetime)
toUnixTimestamp(str, [timezone])

返回值

  • 返回 unix timestamp.

类型: UInt32.

示例

查询:

SELECT toUnixTimestamp('2017-11-05 08:07:47', 'Asia/Tokyo') AS unix_timestamp

结果:

┌─unix_timestamp─┐
│     1509836867 │
└────────────────┘

toStartOfYear

将Date或DateTime向前取整到本年的第一天。 返回Date类型。

toStartOfISOYear

将Date或DateTime向前取整到ISO本年的第一天。 返回Date类型。

toStartOfQuarter

将Date或DateTime向前取整到本季度的第一天。 返回Date类型。

toStartOfMonth

将Date或DateTime向前取整到本月的第一天。 返回Date类型。

!!! 注意 "注意"     解析不正确日期的行为是特定于实现的。 ClickHouse可能会返回零日期抛出异常或执行«natural»溢出。

toMonday

将Date或DateTime向前取整到本周的星期一。 返回Date类型。

toStartOfWeek(t[,mode])

按mode将Date或DateTime向前取整到最近的星期日或星期一。 返回Date类型。 mode参数的工作方式与toWeek()的mode参数完全相同。 对于单参数语法mode使用默认值0。

toStartOfDay

将DateTime向前取整到今天的开始。

toStartOfHour

将DateTime向前取整到当前小时的开始。

toStartOfMinute

将DateTime向前取整到当前分钟的开始。

toStartOfSecond

将DateTime向前取整到当前秒数的开始。

语法

toStartOfSecond(value[, timezone])

参数

  • value — 时间和日期DateTime64.
  • timezone — 返回值的Timezone (可选参数)。 如果未指定将使用 value 参数的时区。 String

返回值

  • 输入值毫秒部分为零。

类型: DateTime64.

示例

不指定时区查询:

WITH toDateTime64('2020-01-01 10:20:30.999', 3) AS dt64
SELECT toStartOfSecond(dt64);

结果:

┌───toStartOfSecond(dt64)─┐
│ 2020-01-01 10:20:30.000 │
└─────────────────────────┘

指定时区查询:

WITH toDateTime64('2020-01-01 10:20:30.999', 3) AS dt64
SELECT toStartOfSecond(dt64, 'Europe/Moscow');

结果:

┌─toStartOfSecond(dt64, 'Europe/Moscow')─┐
│                2020-01-01 13:20:30.000 │
└────────────────────────────────────────┘

参考

toStartOfFiveMinute

将DateTime以五分钟为单位向前取整到最接近的时间点。

toStartOfTenMinutes

将DateTime以十分钟为单位向前取整到最接近的时间点。

toStartOfFifteenMinutes

将DateTime以十五分钟为单位向前取整到最接近的时间点。

toStartOfInterval(time_or_data,间隔x单位[,time_zone])

这是名为toStartOf*的所有函数的通用函数。例如, toStartOfIntervaltINTERVAL 1 year返回与toStartOfYeart相同的结果, toStartOfIntervaltINTERVAL 1 month返回与toStartOfMontht相同的结果, toStartOfIntervaltINTERVAL 1 day返回与toStartOfDayt相同的结果, toStartOfIntervaltINTERVAL 15 minute返回与toStartOfFifteenMinutest相同的结果。

toTime

将DateTime中的日期转换为一个固定的日期同时保留时间部分。

toRelativeYearNum

将Date或DateTime转换为年份的编号从过去的某个固定时间点开始。

toRelativeQuarterNum

将Date或DateTime转换为季度的数字从过去的某个固定时间点开始。

toRelativeMonthNum

将Date或DateTime转换为月份的编号从过去的某个固定时间点开始。

toRelativeWeekNum

将Date或DateTime转换为星期数从过去的某个固定时间点开始。

toRelativeDayNum

将Date或DateTime转换为当天的编号从过去的某个固定时间点开始。

toRelativeHourNum

将DateTime转换为小时数从过去的某个固定时间点开始。

toRelativeMinuteNum

将DateTime转换为分钟数从过去的某个固定时间点开始。

toRelativeSecondNum

将DateTime转换为秒数从过去的某个固定时间点开始。

toISOYear

将Date或DateTime转换为包含ISO年份的UInt16类型的编号。

toISOWeek

将Date或DateTime转换为包含ISO周数的UInt8类型的编号。

toWeek(date[,mode])

返回Date或DateTime的周数。两个参数形式可以指定星期是从星期日还是星期一开始以及返回值应在0到53还是从1到53的范围内。如果省略了mode参数则默认 模式为0。 toISOWeek()是一个兼容函数,等效于toWeek(date,3)。 下表描述了mode参数的工作方式。

Mode First day of week Range Week 1 is the first week …
0 Sunday 0-53 with a Sunday in this year
1 Monday 0-53 with 4 or more days this year
2 Sunday 1-53 with a Sunday in this year
3 Monday 1-53 with 4 or more days this year
4 Sunday 0-53 with 4 or more days this year
5 Monday 0-53 with a Monday in this year
6 Sunday 1-53 with 4 or more days this year
7 Monday 1-53 with a Monday in this year
8 Sunday 1-53 contains January 1
9 Monday 1-53 contains January 1

对于象“with 4 or more days this year,”的mode值根据ISO 86011988对周进行编号

  • 如果包含1月1日的一周在后一年度中有4天或更多天则为第1周。

  • 否则它是上一年的最后一周下周是第1周。

对于像“contains January 1”的mode值, 包含1月1日的那周为本年度的第1周。

toWeek(date, [, mode][, Timezone])

参数

  • date Date 或 DateTime.
  • mode 可选参数, 取值范围 [0,9] 默认0。
  • Timezone 可选参数, 可其他时间日期转换参数的行为一致。

示例

SELECT toDate('2016-12-27') AS date, toWeek(date) AS week0, toWeek(date,1) AS week1, toWeek(date,9) AS week9;
┌───────date─┬─week0─┬─week1─┬─week9─┐
│ 2016-12-27 │    52 │    52 │     1 │
└────────────┴───────┴───────┴───────┘

toYearWeek(date[,mode])

返回Date的年和周。 结果中的年份可能因为Date为该年份的第一周和最后一周而于Date的年份不同。

mode参数的工作方式与toWeek()的mode参数完全相同。 对于单参数语法mode使用默认值0。

toISOYear()是一个兼容函数,等效于intDiv(toYearWeek(date,3),100).

示例

SELECT toDate('2016-12-27') AS date, toYearWeek(date) AS yearWeek0, toYearWeek(date,1) AS yearWeek1, toYearWeek(date,9) AS yearWeek9;
┌───────date─┬─yearWeek0─┬─yearWeek1─┬─yearWeek9─┐
│ 2016-12-27 │    201652 │    201652 │    201701 │
└────────────┴───────────┴───────────┴───────────┘

date_trunc

将Date或DateTime按指定的单位向前取整到最接近的时间点。

语法

date_trunc(unit, value[, timezone])

别名: dateTrunc.

参数

  • unit — 单位. String. 可选值:

    • second
    • minute
    • hour
    • day
    • week
    • month
    • quarter
    • year
  • valueDateTime 或者 DateTime64.

  • timezoneTimezone name 返回值的时区(可选值)。如果未指定将使用value的时区。 String.

返回值

  • 按指定的单位向前取整后的DateTime。

类型: Datetime.

示例

不指定时区查询:

SELECT now(), date_trunc('hour', now());

结果:

┌───────────────now()─┬─date_trunc('hour', now())─┐
│ 2020-09-28 10:40:45 │       2020-09-28 10:00:00 │
└─────────────────────┴───────────────────────────┘

指定时区查询:

SELECT now(), date_trunc('hour', now(), 'Europe/Moscow');

结果:

┌───────────────now()─┬─date_trunc('hour', now(), 'Europe/Moscow')─┐
│ 2020-09-28 10:46:26 │                        2020-09-28 13:00:00 │
└─────────────────────┴────────────────────────────────────────────┘

参考

now

返回当前日期和时间。

语法

now([timezone])

参数

返回值

  • 当前日期和时间。

类型: Datetime.

示例

不指定时区查询:

SELECT now();

结果:

┌───────────────now()─┐
│ 2020-10-17 07:42:09 │
└─────────────────────┘

指定时区查询:

SELECT now('Europe/Moscow');

结果:

┌─now('Europe/Moscow')─┐
│  2020-10-17 10:42:23 │
└──────────────────────┘

today

不接受任何参数并在请求执行时的某一刻返回当前日期(Date)。 其功能与toDatenow())’相同。

yesterday

不接受任何参数并在请求执行时的某一刻返回昨天的日期(Date)。 其功能与today() - 1相同。

timeSlot

将时间向前取整半小时。 此功能用于Yandex.Metrica因为如果跟踪标记显示单个用户的连续综合浏览量在时间上严格超过此数量则半小时是将会话分成两个会话的最短时间。这意味着tag iduser idtime slot可用于搜索相应会话中包含的综合浏览量。

toYYYMM

将Date或DateTime转换为包含年份和月份编号的UInt32类型的数字YYYY * 100 + MM

toYYYMMDD

将Date或DateTime转换为包含年份和月份编号的UInt32类型的数字YYYY * 10000 + MM * 100 + DD

toYYYYMMDDhhmmss

将Date或DateTime转换为包含年份和月份编号的UInt64类型的数字YYYY * 10000000000 + MM * 100000000 + DD * 1000000 + hh * 10000 + mm * 100 + ss

addYears, addMonths, addWeeks, addDays, addHours, addMinutes, addSeconds, addQuarters

函数将一段时间间隔添加到Date/DateTime然后返回Date/DateTime。例如

WITH
    toDate('2018-01-01') AS date,
    toDateTime('2018-01-01 00:00:00') AS date_time
SELECT
    addYears(date, 1) AS add_years_with_date,
    addYears(date_time, 1) AS add_years_with_date_time
┌─add_years_with_date─┬─add_years_with_date_time─┐
│          2019-01-01 │      2019-01-01 00:00:00 │
└─────────────────────┴──────────────────────────┘

subtractYears,subtractMonths,subtractWeeks,subtractDays,subtractours,subtractMinutes,subtractSeconds,subtractQuarters

函数将Date/DateTime减去一段时间间隔然后返回Date/DateTime。例如

WITH
    toDate('2019-01-01') AS date,
    toDateTime('2019-01-01 00:00:00') AS date_time
SELECT
    subtractYears(date, 1) AS subtract_years_with_date,
    subtractYears(date_time, 1) AS subtract_years_with_date_time
┌─subtract_years_with_date─┬─subtract_years_with_date_time─┐
│               2018-01-01 │           2018-01-01 00:00:00 │
└──────────────────────────┴───────────────────────────────┘

dateDiff

返回两个Date或DateTime类型之间的时差。

语法

dateDiff('unit', startdate, enddate, [timezone])

参数

  • unit — 返回结果的时间单位。 String.

    支持的时间单位: second, minute, hour, day, week, month, quarter, year.
    
  • startdate — 第一个待比较值。 DateDateTime.

  • enddate — 第二个待比较值。 DateDateTime.

  • timezone — 可选参数。 如果指定了,则同时适用于startdateenddate。如果不指定,则使用startdateenddate的时区。如果两个时区不一致,则结果不可预料。

返回值

unit为单位的startdateenddate之间的时差。

类型: int.

示例

查询:

SELECT dateDiff('hour', toDateTime('2018-01-01 22:00:00'), toDateTime('2018-01-02 23:00:00'));

结果:

┌─dateDiff('hour', toDateTime('2018-01-01 22:00:00'), toDateTime('2018-01-02 23:00:00'))─┐
│                                                                                     25 │
└────────────────────────────────────────────────────────────────────────────────────────┘

timeSlots(StartTime, Duration,[, Size])

它返回一个时间数组其中包括从从«StartTime»开始到«StartTime + Duration 秒»内的所有符合«size»以秒为单位步长的时间点。其中«size»是一个可选参数默认为1800。 例如,timeSlots(toDateTime('2012-01-01 12:20:00')600) = [toDateTime'2012-01-01 12:00:00'toDateTime'2012-01-01 12:30:00' ]。 这对于搜索在相应会话中综合浏览量是非常有用的。

formatDateTime

函数根据给定的格式字符串来格式化时间。请注意:格式字符串必须是常量表达式,例如:单个结果列不能有多种格式字符串。

语法

formatDateTime(Time, Format\[, Timezone\])

返回值

根据指定格式返回的日期和时间。

支持的格式修饰符

使用格式修饰符来指定结果字符串的样式。«Example» 列是对2018-01-02 22:33:44的格式化结果。

修饰符 描述 示例
%C 年除以100并截断为整数(00-99) 20
%d 月中的一天零填充01-31) 02
%D 短MM/DD/YY日期相当于%m/%d/%y 01/02/2018
%e 月中的一天,空格填充( 1-31) 2
%F 短YYYY-MM-DD日期相当于%Y-%m-%d 2018-01-02
%G ISO周号的四位数年份格式 从基于周的年份由ISO 8601定义 标准计算得出通常仅对V有用 2018
%g 两位数的年份格式与ISO 8601一致四位数表示法的缩写 18
%H 24小时格式00-23) 22
%I 12小时格式01-12) 10
%j 一年中的一天 (001-366) 002
%m 月份为十进制数01-12) 01
%M 分钟(00-59) 33
%n 换行符(")
%p AM或PM指定 PM
%Q 季度1-4) 1
%R 24小时HH:MM时间相当于%H:%M 22:33
%S 秒 (00-59) 44
%t 水平制表符()
%T ISO8601时间格式(HH:MM:SS),相当于%H:%M:%S 22:33:44
%u ISO8601工作日为数字星期一为1(1-7) 2
%V ISO8601周编号(01-53) 01
%w 工作日为十进制数周日为0(0-6) 2
%y 年份最后两位数字00-99) 18
%Y 2018
%% %符号 %

示例

查询:

SELECT formatDateTime(toDate('2010-01-04'), '%g')

结果:

┌─formatDateTime(toDate('2010-01-04'), '%g')─┐
│ 10                                         │
└────────────────────────────────────────────┘

Original article

FROM_UNIXTIME

当只有单个整数类型的参数时,它的作用与toDateTime相同,并返回DateTime类型。

例如:

SELECT FROM_UNIXTIME(423543535)
┌─FROM_UNIXTIME(423543535)─┐
│      1983-06-04 10:58:55 │
└──────────────────────────┘

当有两个参数时第一个是整型或DateTime第二个是常量格式字符串它的作用与formatDateTime相同,并返回String类型。

例如:

SELECT FROM_UNIXTIME(1234334543, '%Y-%m-%d %R:%S') AS DateTime
┌─DateTime────────────┐
│ 2009-02-11 14:42:23 │
└─────────────────────┘