Error updating database. Cause: java.lang.RuntimeException: Clock moved backwards.

java.lang.RuntimeException: Clock moved backwards. Refusing to generate id for 2590445467 milliseconds。

这个异常通常发生在使用Java的System.currentTimeMillis()函数来生成ID或在数据库事务中使用时间戳的情况下。当系统时钟被手动或自动(例如,夏令时调整)向后调整时,就会出现这个问题。

这是因为在Java中,System.currentTimeMillis()返回的是从1970年1月1日00:00:00 UTC开始的毫秒数。如果系统时钟被调整回之前的时间,那么System.currentTimeMillis()就会返回一个之前已经返回过的值,这就可能导致ID的重复生成或者在数据库事务中出现时间戳混乱的情况。

解决这个问题的方法主要有两种:

  1. 调整系统时钟:确保你的系统时钟没有被向后调整。如果你在一个需要频繁调整系统时钟的环境中(例如,经常跨越夏令时边界),你可能需要考虑使用一个更稳定的时钟源,或者使用一个不受系统时钟影响的ID生成方法。
  2. 更改ID生成策略:不要依赖System.currentTimeMillis()来生成ID。你可以使用更复杂的ID生成策略,例如UUID,或者使用数据库的自增ID。这些策略都不会受到系统时钟调整的影响。

对于数据库更新失败的问题,可能是因为你在事务中使用了系统时间戳,当系统时钟调整后,时间戳变小,导致数据库认为事务的时间戳不正确,从而拒绝了更新。你可以尝试使用数据库的自增ID或者其他不受系统时钟影响的策略来解决这个问题。

本次解决方案:重启所有服务,单独重启一个未解决

public synchronized String nextId() { long timestamp = timeGen(); //获取当前毫秒数 //如果服务器时间有问题(时钟后退) 报错。 if (timestamp < lastTimestamp) { throw new RuntimeException(String.format( "Clock moved backwards. Refusing to generate id for %d milliseconds", lastTimestamp - timestamp)); } //如果上次生成时间和当前时间相同,在同一毫秒内 if (lastTimestamp == timestamp) { //sequence自增,因为sequence只有12bit,所以和sequenceMask相与一下,去掉高位 sequence = (sequence + 1) & sequenceMask; //判断是否溢出,也就是每毫秒内超过4095,当为4096时,与sequenceMask相与,sequence就等于0 if (sequence == 0) { timestamp = tilNextMillis(lastTimestamp); //自旋等待到下一毫秒 } } else { sequence = 0L; //如果和上次生成时间不同,重置sequence,就是下一毫秒开始,sequence计数重新从0开始累加 } lastTimestamp = timestamp; long suffix = (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence; String datePrefix = DateFormatUtils.format(timestamp, "yyyyMMddHHMMssSSS"); return datePrefix + suffix; } protected long tilNextMillis(long lastTimestamp) { long timestamp = timeGen(); while (timestamp <= lastTimestamp) { timestamp = timeGen(); } return timestamp; } protected long timeGen() { return System.currentTimeMillis(); } private byte getLastIP(){ byte lastip = 0; try{ InetAddress ip = InetAddress.getLocalHost(); byte[] ipByte = ip.getAddress(); lastip = ipByte[ipByte.length - 1]; } catch (UnknownHostException e) { e.printStackTrace(); } return lastip; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

大小先生

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值