现在,我的理解是,当 start() 被调用时,JVM 隐式(并立即)调用 run() 方法......
这是不正确的。它确实隐式调用run()
,但调用不一定立即发生。
现实情况是,新线程可以在start()
调用后的某个时间点进行调度。实际调度取决于本机调度程序。它可能会立即发生,或者父线程可能会在子线程被调度之前持续一段时间。
要强制您的线程立即开始运行(或者更准确地说,在之前开始运行doSomethingElse()
),您需要进行一些显式同步;例如这样的:
java.util.concurrent.CountDownLatch latch = new CountdownLatch(1);
new Thread(new MyRunnable(latch)).start();
latch.await(); // waits until released by the child thread.
doSomethingElse();
在哪里
class MyRunnable implements Runnable {
private CountDownLatch latch;
MyRunnable (CountDownLatch latch) { this.latch = latch; }
public void run() {
doSomeStuff();
latch.countDown(); // releases the parent thread
doSomeMoreStuff();
}
...
}
还有其他方法可以使用并发类或 Java 的 mutex/wait/notify 原语1来实现同步。但是两个线程之间的显式同步是保证您需要的行为的唯一方法。
请注意,子线程中的调用会在父线程释放之前完成,但对于anddoSomething()
的执行顺序我们无话可说。(一个可能在另一个之前运行,反之亦然,或者它们可能并行运行。)doSomethingElese()
doSomeMoreStuff()
1 - 不推荐使用wait
/ notify
,但如果并发 API 不可用,它可能是您唯一的选择;例如在 Java ME 上。