ClickHouse使用PreparedStatement占位符批量插入 字段类型Float64 某些数值精度缺失问题解决

项目场景:

在做离线文件导入ck的需求,其中带小数的使用float64来存储(之前是decimal,由于业务不得不调整).


问题描述

name 使用string类型来存储
core 使用float64类型来存储
代码使用 insert into table (name, core) values (?, ?) (?, ?) 占位符方式,再使用PreparedStatement.setObject(int parameterIndex, Object x) 方法替换占位符. 在使用的时候 float64如果是1.852 很神奇落库会失真变成1.8519999999999999.具体原因需要深入研究ck.

解决方案:

  1. 使用Statement,通过拼接的方式将float类型的数值用toDecimal()函数包一下: toDecimal64(1.852,4) 这里保留4位小数;
insert into table (name, core) values ('zhangsan', toDecimal64(1.852,4)) ('lisi', toDecimal64(1.854,4)) 

问题出在这里,使用批量插入调用下面方法耗时10秒左右(测试5000条20多个字段),这使得交互很慢

statement.addBatch(sql);

从网上找到的说法
在这里插入图片描述
2. 还是使用PreparedStatement, 不能在数值赋值出拼接toDecimal64(1.854,4), 因为setObject方法会解析成字符串. 只要在问号拼接处将toDecimal64()拼接上就可以解决这个问题.我这里是知道字段类型的情况下做逻辑区分

insert into table (name, core) values  (?, toDecimal64(?,4)) (?, toDecimal64(?,4))  

这个时候使用setObject方法就可以完美解决精度缺失问题

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值