更改操作系统时间时sleep()中的Java错误:有任何解决方法?
使我烦恼的臭虫和这张票一样。基本上,如果将操作系统时钟更改为过去的日期,则更改时休眠的所有线程都不会唤醒。
我正在开发的应用程序打算在24/24上运行,我们希望能够在不停止OS的情况下更改OS日期(例如,从夏季时间切换为冬季时间)。目前发生的事情是,当我们将日期更改为过去的日期时,应用程序的某些部分就会冻结。我在多台计算机上,Windows
XP和Linux 2.6.37以及最近的JVM(1.6.0.22)上都观察到了这一点。
我尝试了许多Java睡眠原语,但是它们都有相同的行为:
- Thread.sleep(长)
- Thread.sleep(long,int)
- Object.wait(长)
- Object.wait(long,int)
- Thread.join(长)
- Thread.join(long,int)
- LockSupport.parkNanos(长)
- java.util.Timer
- javax.swing.Timer
现在,我不知道要解决此问题。我认为我无法采取任何措施来防止睡眠线程冻结。但是,我至少希望在检测到危险的系统时钟更改时向用户发出警告。
我想出了一个监视线程来检测这种变化:
Thread t = new Thread(new Runnable() { @Override
public void run() {
long ms1 = System.currentTimeMillis();
long ms2;
while(true) {
ms2 = ms1;
ms1 = System.currentTimeMillis();
if (ms1 < ms2) {
warnUserOfPotentialFreeze();
}
Thread.yield();
}
}
});
t.setName("clock monitor");
t.setPriority(Thread.MIN_PRIORITY);
t.setDaemon(true);
t.start();
问题在于,这会使应用程序在空闲时从2%的CPU使用率增长到15%。
您是否有解决原始问题的想法,还是可以想到另一种监视线程冻结外观的方法?
编辑
Ingo建议不要触摸系统时钟。我同意通常不需要。问题在于我们无法控制客户使用计算机的行为(我们计划出售数百份)。
更糟糕:我们的一台机器没有任何人工干预就出现此问题。我猜想操作系统(Windows
XP)会定期将其时钟与RTC时钟同步,这会使操作系统时钟自然地回到时间上。
结语
我发现问题中的某些陈述是错误的。实际上,我的最初问题涉及两个单独的原因。现在,我可以肯定地说两件事:
在我的机器只(ArchLinux的内核2.6.37与OpenJDK的64位1.6.0_22) ,
Thread.sleep
,,Object.wait
有同样的问题:他们醒来,只有当系统时钟到达觉醒的“目标”的时间。但是,在我的shell中没有一个简单的问题。Thread.join``LockSupport.parkNanos``sleep
在所有的机器我测试(包括矿山),
java.util.Timer
并java.swing.Timer
有同样的问题(他们被封锁,直到达到“目标”时间)。
因此,我所做的是,我用Timer
一个更简单的实现替换了所有Java 。这解决了除我之外的所有机器的问题(我只是希望我的机器是一个例外,而不是一条规则)。
回答:
根据错误通知单,您的线程不会冻结,一旦时钟赶上修改之前的时间,它们将恢复运行(因此,如果将其移回一小时,则您的线程将在1小时内恢复运行)。
当然,这仍然不是很有用。根本原因似乎是Thread.sleep()
解决一个系统调用,该调用使线程进入睡眠状态,直到将来某个特定的时间戳记为止,而不是持续指定的持续时间。要解决此问题,您需要实现自己Thread.sleep()
使用的版本,System.nanoTime()
而不是使用System.currentTimeMillis()
API或任何其他依赖时间的API。Thread.sleep()
但是,我无法在不使用内置功能的情况下做到这一点。
编辑:
或者,如果您用另一种语言(例如C或您喜欢的其他语言)创建某个外部应用程序,该应用程序除了等待指定的时间后什么都不做,然后退出,该怎么办?然后,您可以生成此外部进程的新实例,然后在其上调用waitFor(),而不是在Java中调用Thread.sleep()。这将使Java线程出于所有实际目的“休眠”,并且只要您的外部应用程序能够以正确的时间休眠,它就会在正确的时间恢复运行而不会冻结,也不会影响CPU。
解决这个问题似乎还有很长的路要走,但这是我能想到的唯一可行的解决方法。同样,由于产生外部进程是一个相对昂贵的操作,因此,如果您睡眠相对较长的时间(例如几百毫秒或更长),则效果最好。在较短的时间内,它可能只会继续影响CPU。
以上是 更改操作系统时间时sleep()中的Java错误:有任何解决方法? 的全部内容, 来源链接: utcz.com/qa/418709.html