作者:总裁班的草根 | 来源:互联网 | 2024-09-28 16:37
我正在研究线程,并决定在运行我的重点代码行之前和之后添加一些额外的文本,以供参考。我期望在开始时获得一个“额外文本”,而在结束时获得另一个。但是...没有发生,当我运行它时,第二个“额外文本”仅位于第四位置。我是初学者,需要知道为什么会这样...
--- CODE ---
class Hi extends Thread{
public void run(){
for(int i=1; i<=5; i++){
System.out.println("HI!");
try{
Thread.sleep(500);
} catch(InterruptedException e){}
}
}
}
class Hey extends Thread{
public void run(){
for(int i=1; i<=5; i++){
System.out.println("HEY!");
try{
Thread.sleep(500);
} catch(InterruptedException e){}
}
}
}
public class MyClass {
public static void main(String[] args){
Hi hiObj = new Hi();
Hey heyObj = new Hey();
System.out.println("extra-text");
hiObj.start();
heyObj.start();
try {
Thread.currentThread().sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("extra-text");
}
}
---输出---
extra-text
HI!
HEY!
extra-text
HEY!
HI!
HEY!
HI!
HEY!
HI!
HEY!
HI!
这是常见的并发错误。
程序的主方法在主线程上运行。因此,在启动hiObj
和heyObj
线程之前,您已经有一个线程。启动两个新线程后,您将拥有三个。每个都同时执行。这意味着每个线程可以执行代码而无需等待其他线程。不能保证线程之间的顺序。
这会导致您观察到的行为。在启动hiObj
或heyObj
之前,在主线程上运行的main方法将打印"extra-text"
。接下来,启动hiObj和heyObj。主线程到达行Thread.currentThead().sleep(10)
,这导致它暂停执行10毫秒。在大多数计算机(包括您的计算机)上,这足以让其他两个线程开始执行。每个线程都以其for
方法开始run
循环,并打印"HI"
或"HEY"
。因此,输出的前三行是(不保证"HI"
和"HEY"
的顺序):
"extra-text"
"HI"
"HEY"
接下来,hiObj
和heyObj
线程到达行Thread.sleep(500)
,这使它们暂停执行500毫秒。经过10毫秒后,主线程将完成睡眠并恢复运行。请注意,hiObj
或heyObj
线程现在都无法恢复。因此,打印的下一行将是在main
中执行的下一行的。这是"extra-text"
。因此,预期输出为:
"extra-text"
"HI"
"HEY"
"extra-text"
在接下来的几秒钟内,将发生hiObj
和heyObj
线程中的其余打印。在Java中,主线程仅在所有其他线程都退出之后退出(除非调用System.exit
或存在未捕获的异常)。在这种情况下,这意味着程序仅在main
到达执行结束时以及hiObj
和heyObj
的run方法都返回时退出。
要更改程序,以使最后的"extra-text"
始终显示在末尾,则必须使主线程等待hiObj
和heyObj
线程完成。在Java中,Thead
上有一个名为join
的方法,该方法使调用线程等待,直到连接的线程死亡。在程序中,您可以将MyClass
修改为如下形式:
public class MyClass {
public static void main(String[] args){
Hi hiObj = new Hi();
Hey heyObj = new Hey();
System.out.println("extra-text");
hiObj.start();
heyObj.start();
try {
Thread.currentThread().sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
hiObj.join();
heyObj.join();
System.out.println("extra-text");
}
}
进行此更改后,main
将首先等待hiObj
完成,然后等待heyObj
完成,然后再打印"extra-text"
。
,
如果您摆脱
Thread.currentThread().sleep(10);
在主方法中,您将看到执行后立即将另外两个文本打印到控制台。通过使用sleep(10),您只需延迟第二个额外的文本,与此同时,您的2个线程将打印其第一个输出。