更改操作系统时间时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时钟同步,这会使操作系统时钟自然地回到时间上。

结语

我发现问题中的某些陈述是错误的。实际上,我的最初问题涉及两个单独的原因。现在,我可以肯定地说两件事:

  1. 在我的机器只(ArchLinux的内核2.6.37与OpenJDK的64位1.6.0_22) ,Thread.sleep,,Object.wait 有同样的问题:他们醒来,只有当系统时钟到达觉醒的“目标”的时间。但是,在我的shell中没有一个简单的问题。Thread.join``LockSupport.parkNanos``sleep

  2. 在所有的机器我测试(包括矿山),java.util.Timerjava.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

回到顶部