类型转换
Append/AppendRow) 和读取时 (通过 Scan) 两部分。如果你需要支持某种特定转换,请提交 issue。
标准的 database/sql 接口应支持与 ClickHouse API 相同的类型。也有少数例外,主要涉及复杂类型,相关内容已在下文各节中说明。与 ClickHouse API 类似,客户端在接受用于插入和响应编组的变量类型方面,也力求尽可能灵活。
复杂类型
Date/DateTime
Date、Date32、DateTime 和 DateTime64 日期/日期时间类型。日期既可以以 2006-01-02 格式的字符串插入,也可以使用原生 Go 类型 time.Time{} 或 sql.NullTime。DateTime 同样支持后两种类型,但字符串必须采用 2006-01-02 15:04:05 格式,并可选择附带时区偏移,例如 2006-01-02 15:04:05 +08:00。读取时也支持 time.Time{} 和 sql.NullTime,以及任何实现了 sql.Scanner 接口的类型。
时区信息的处理方式取决于 ClickHouse 类型,以及该值是被插入还是读取:
- DateTime/DateTime64
- 在 insert 时,值会以 UNIX timestamp 格式发送到 ClickHouse。如果未提供时区,客户端会假定使用客户端本地时区。
time.Time{}或sql.NullTime也会据此转换为自 Unix 纪元起的时间戳。 - 在 select 时,返回
time.Time值时,如果列设置了 timezone,则使用列的 timezone;否则使用 server 的 timezone。
- 在 insert 时,值会以 UNIX timestamp 格式发送到 ClickHouse。如果未提供时区,客户端会假定使用客户端本地时区。
- Date/Date32
- 在 insert 时,将日期转换为 unix timestamp 时会考虑日期的 timezone;也就是说,在按日期存储之前,会先根据 timezone 进行偏移,因为 ClickHouse 中的 Date 类型本身不包含时区信息。如果字符串值中未指定这一信息,则会使用本地 timezone。
- 在 select 时,扫描到
time.Time{}或sql.NullTime{}实例中的日期值,返回时将不包含时区信息。
Time/Time64 类型
Time 和 Time64 列类型用于存储不包含日期部分的时刻值。两者都映射为 Go 的 time.Duration。
Time以秒级精度存储时刻。Time64(precision)支持子秒级精度 (类似于DateTime64) ,其中 precision 的取值范围为 0–9。
Array
Map
使用 database/sql API 时,Map 值必须严格指定类型,不能将
interface{} 用作值类型。例如,对于 Map(String,String) 字段,不能传入 map[string]interface{},而必须使用 map[string]string。不过,interface{} 类型的变量始终是兼容的,因此可用于更复杂的结构。完整示例元组
Nested
map[string]interface{}。这些值目前没有强类型约束。
flatten_tested=0
如果 flatten_nested 使用默认值 1,嵌套列会被展平为独立的数组。因此,在插入和读取时需要使用嵌套切片。虽然任意级别的嵌套可能也可以工作,但这并不属于官方支持的范围。
flatten_nested=1
注意:Nested 列中的各列必须具有相同的维度。例如,在上面的示例中,Col_2_2 和 Col_2_1 必须包含相同数量的元素。
由于其接口更直观,且官方支持嵌套,我们建议使用 flatten_nested=0。
Geo 类型
UUID
sql.Scanner 或 Stringify 的类型进行发送和序列化。
Decimal
你可能会想改用 Float,以避免引入第三方依赖。不过请注意,在需要精确值时,不建议在 ClickHouse 中使用 Float 类型。如果你仍然选择在客户端使用 Go 的内置 Float 类型,则必须在 ClickHouse 查询中使用 toFloat64() 函数 或 其变体 显式将 Decimal 转换为 Float。请注意,这种转换可能会导致精度损失。
Nullable
sql.Null* 类型,例如 sql.NullInt64。这些类型与对应的 ClickHouse 类型兼容。
大整数
BFloat16
BFloat16 是一种用于机器学习场景的 16 位 brain float 类型。在 Go 中,BFloat16 值以 float32 形式插入和扫描。Nullable 变体使用 sql.NullFloat64。
QBit
QBit 是一种 Experimental 列类型,用于以位切片格式存储嵌入向量,并针对向量相似性搜索进行了优化。它要求启用 allow_experimental_qbit_type 设置。
在 Go 中,QBit(Float32, N) 列会以 []float32 的形式插入和扫描,其中 N 为向量维度。