java多线程之join与yield

来源:互联网 发布:windows杀死进程命令 编辑:程序博客网 时间:2024/05/21 18:44

转载请注明出处

http://blog.csdn.net/pony_maggie/article/details/43897971

作者:小马

 

先说说join的用法, 在某些情况下,如果子线程里要进行大量的耗时的运算,主线程可能会在子线程执行完之前结束,但是如果主线程又需要用到子线程的处理结果,也就是主线程需要等待子线程执行完成之后再结束,这个时候就要用到join()。

 

看代码示例,

[java] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. class Thread1 extends Thread  
  2. {  
  3.     public Thread1()  
  4.     {  
  5.         super("[Thread1] Thread");  
  6.     }  
  7.     public void run()  
  8.     {  
  9.         String threadName = Thread.currentThread().getName();  
  10.         System.out.println(threadName + " start.");  
  11.         try   
  12.         {  
  13.             for(int i = 0; i < 5; i++)  
  14.             {  
  15.                 System.out.println(threadName + " loop at " + i);  
  16.                 Thread.sleep(1000);  
  17.             }  
  18.             System.out.println(threadName + " end.");  
  19.         }   
  20.         catch (Exception e)   
  21.         {  
  22.             System.out.println("exception from " + threadName + ".run");  
  23.         }  
  24.           
  25.           
  26.     }  
  27. }  
  28.   
  29. class Thread2 extends Thread  
  30. {  
  31.     Thread1 t1;  
  32.     public Thread2(Thread1 thread1)  
  33.     {  
  34.         super("[Thread2] Thread");  
  35.         this.t1 = thread1;  
  36.     }  
  37.     public void run()  
  38.     {  
  39.         String threadName = Thread.currentThread().getName();  
  40.         System.out.println(threadName + " start.");  
  41.         try   
  42.         {  
  43.             t1.join();  
  44.             System.out.println(threadName + " end.");  
  45.         }   
  46.         catch (Exception e)   
  47.         {  
  48.             System.out.println("exception from" + threadName + ".run");  
  49.         }  
  50.           
  51.           
  52.     }  
  53. }  
  54.   
  55.   
  56. public class JoinAndYieldTestDemo   
  57. {  
  58.   
  59.     /** 
  60.      * @param args 
  61.      */  
  62.     public static void main(String[] args)   
  63.     {  
  64.   
  65.         String threadName = Thread.currentThread().getName();  
  66.         System.out.println(threadName + " start.");  
  67.         Thread1 t1 = new Thread1();  
  68.         Thread2 t2 = new Thread2(t1);  
  69.         try   
  70.         {  
  71.             t1.start();  
  72.             Thread.sleep(2000);  
  73.             t2.start();  
  74.             t2.join();  
  75.         } catch (Exception e)   
  76.         {  
  77.             System.out.println("exception from main");  
  78.         }  
  79.           
  80.         System.out.println(threadName + " end.");  
  81.       
  82.     }  
  83.   
  84. }  


 

运行结果:

[html] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. main start.    //主线程起动   
  2. [Thread1] Thread start.    
  3. [Thread1] Thread loop at 0    
  4. [Thread1] Thread loop at 1    
  5. [Thread2] Thread start.       
  6. [Thread1] Thread loop at 2    
  7. [Thread1] Thread loop at 3    
  8. [Thread1] Thread loop at 4    
  9. [Thread1] Thread end.    
  10. [Thread2] Thread end.        
  11. main end!        



从运行结果分析,t2一定在t1之后才结束,main线程则一定在t2结束之后结束,这都要归功于各自调用的join方法

再来说说yield,建议先看看前一篇文章,关于线程优先级,
http://blog.csdn.net/pony_maggie/article/details/43889225

 

理论上虚拟机和操作系统会让优先级高的的线程更多的获取执行的机会,这个不要与时间片混淆。yield的意思是放手,放弃,一个线程调用yield意味着它要告诉虚拟机自己乐意让其它线程占用自己的位置。这只是一个暗示,并不保证会产生效果(使当前线程转到可运行状态,runnable)。

 

看代码示例,

[java] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. class Producer extends Thread  
  2. {  
  3.     public void run()  
  4.     {  
  5.         for(int i = 0; i < 5; i++)  
  6.         {  
  7.             System.out.println("i am producer: producerd item " + i);  
  8.             Thread.yield();  
  9.         }  
  10.     }  
  11. }  
  12.   
  13. class Consumer extends Thread  
  14. {  
  15.     public void run()  
  16.     {  
  17.         for(int i = 0; i < 5; i++)  
  18.         {  
  19.             System.out.println("i am Consumer: Consumed item " + i);  
  20.             Thread.yield();  
  21.         }  
  22.     }  
  23. }  
  24. public class JoinAndYieldTestDemo   
  25. {  
  26.   
  27.     /** 
  28.      * @param args 
  29.      */  
  30.     public static void main(String[] args)   
  31.     {  
  32.           
  33.         Thread producer = new Producer();  
  34.         Thread consumer = new Consumer();  
  35.           
  36.         producer.setPriority(Thread.MIN_PRIORITY);  
  37.         consumer.setPriority(Thread.MAX_PRIORITY);  
  38.           
  39.         producer.start();  
  40.         consumer.start();  
  41.     }  
  42.   
  43. }  



程序创建了名为生产者和消费者的两个线程,前者最小优先级,后者最大优先级。调用yield方法,两个线程会依次打印,然后将执行机会交给对方,一直这样进行下去。如果不加yield,结果是先打印完一个,再是另一个。

 

yield的作用一目了然,设置成不同的优先级,也证明了yield不会受优先级大小的影响。

程序中的yield()函数简单说几句,其实和sleep功能相似,都是把当前线程暂停,这样其它线程才能有机会执行。区别就是它没有参数,不能像sleep那样指定暂停的时间。

还有个区别很重要,yield只能让同级别优先级的线程得到执行机会。sleep不分优先级,只有当前线程sleep了,其它线程就有机会执行。否则没有sleep的话,只能等高优先级执行完毕,才能执行低优先级的线程。

说了这么多,似乎并没有说到daemon线程的优点,哪些情况要用到它呢?举个例子,比如垃圾回收线程,我们希望用户线程存在时,它也存在,但当所有的用户线程退出时,程序就要退出,垃圾回收线程不要影响退出。如果定义成了用户线程,那么只要垃圾回收线程不退出,用户线程就不会退出,与实际需求不相符

0 0
原创粉丝点击