背景
我想了解为什么一段代码不会抛出NullPointerException.
源代码
请考虑以下代码:
public class Agent { public List files = new ArrayList(); public void deliver() { if( files != null && files.iterator().hasNext() ) { File file = (File)files.iterator().next(); } files = new ArrayList(); } }
public void run() { agent.files = null; }
只有一个代理程序实例.
问题
永远不会抛出NullPointerException.
但是,当传递方法暂停时,即使是0毫秒,也会按预期抛出NullPointerException:
public void deliver() { if( files != null ) { Thread.currentThread().sleep( 0 ); if( files.iterator().hasNext() ) { File file = (File)files.iterator().next(); } } files = new ArrayList(); }
我的理解是,理论上,在检查文件== null和调用files.iterator().hasNext()之间存在竞争条件.在实践中,我不能在不引入暂停的情况下触发竞争条件(即,从后续方法调用中拆分空检查).
题
为什么第一个传递方法在同一语句中组合空检查和使用时不会抛出异常?
解决方法
两件事情:
> Thread.sleep(0)仍然停止执行(可能超过0毫秒).基本上,即使是0睡眠也会导致该线程上的执行暂时停止,然后重新启动.这使得另一个线程有机会运行和完成,这就是为什么你能够触发竞争条件.>文件应该是易失性的,否则允许JVM以这样的方式进行优化,使得您可能永远不会注意到它正在改变值,因为它不认为它需要保持线程之间的一致性.