多线程细节

1、sleep和wait方法的异同

  sleep和wait都可以让线程睡眠

  不同:

    sleep需要指定时间,而wait可以指定时间,也可以不用指定时间。

    sleep时间到了线程就会重新恢复到运行状态。wait如果没有指定时间,必须通过notify、notifyAll方法来唤醒

    sleep可以使用同步代码块中,也可以不使用同步代码块中,wait必须使用同步代码块中。

    sleep睡眠后不会释放锁,而wait会释放锁。

2、停止线程和interrupt方法

  一般情况下线程要结束,都是通过run方法运行完成,线程就结束了。如果要线程提前中止运行,可以使用stop方法。但是stop方法过时了,原因有不安全因素。stop方法中给出了相应的解决方案:在线程中可以定义一些变量或者标记,然后在线程运行的期间,判断这个变量或者标记,最终让线程执行的仍然运行完成。如果线程长时间处于等待(冻结)状态可以使用interrupt方法来中断这种等待。

  interrupt()  其实是把线程这种等待状态给清除,让线程重新恢复到运行状态。

 1 //测试线程停止和中断等待
 2 
 3 class Demo implements Runnable
 4 {
 5     boolean flag = false;
 6     public void run()
 7     {
 8         while( true )
 9         {
10             synchronized( this )
11             {
12                 System.out.println(Thread.currentThread().getName()+"......"+flag);
13 
14                 try{ this.wait(); }catch(Exception e){}
15                 
16                 if(flag)
17                 {
18                     break;
19                 }
20             }
21         }
22     
23     }
24 }
25 class ThreadDemo 
26 {
27     public static void main(String[] args) 
28     {
29         Demo d = new Demo();
30 
31         Thread t = new Thread(d);
32         Thread t2 = new Thread(d);
33 
34         t.start();
35         t2.start();
36 
37 
38         for( int i=0;i<100;i++ )
39         {
40             System.out.println(Thread.currentThread().getName()+"========"+i);
41             if( i==50 )
42             {
43                 d.flag = true;
44                 //清除Thread-0的等待状态
45                 t.interrupt();
46                 t2.interrupt();
47             }
48         }
49     }
50 }

3、守护线程、线程组、线程优先级

  守护线程:后台线程。当我们直接创建的线程,没有设置它的任何属性时,这个线程一般都属于非守护线程(前台线程)。

  非守护线程,它和主线程一起执行,当主线程执行完成之后,如果还有其他非守护线程没有执行结束,当前程序依然在正常的运行。

  守护线程:它也是用来执行线程任务的,它会随着非守护线程一起执行。当程序中没有任何一个非守护线程在运行时,这时不管有多少个守护线程,它们及时没有执行完成,程序也会自动停止运行。守护线程就自动结束运行了。

 

  线程组:把完成类似功能的线程可以合并在一起,使用一个组来管理当前这些线程,即可以通过组的方式来管理线程。

 

  线程优先级:线程优先级采用数字来标识,从1开始到10,一般情况下会使用1   5   10  来标记当前的线程的优先级

优先级高的线程执行的次数会比优先级低的线程多。优先级高,只能让cpu执行到这个线程的几率变高,而不是说cpu就一直在执行它。

getPriority()   获取当前线程的优先级

setPriority(int newPriority)  设置线程的优先级,newPriority参数可以是1-10!建议设置的时候1  5  10 这三个数字。

  守护线程源码:

 1 //测试线程停止和中断等待
 2 
 3 class Demo implements Runnable
 4 {
 5     boolean flag = false;
 6     public void run()
 7     {
 8         int num = 1;
 9         while( num<20 )
10         {
11             System.out.println(Thread.currentThread().getName()+"......"+num++);
12         }
13     }
14 }
15 class ThreadDemo2 
16 {
17     public static void main(String[] args) throws InterruptedException
18     {
19         Demo d = new Demo();
20 
21         Thread t = new Thread(d);
22         Thread t2 = new Thread(d);
23         t.setPriority(10);
24         t2.setPriority(1); 
25 
26         //把2个线程都设置成守护线程
27         t.setDaemon(true);
28         t2.setDaemon(true);
29         t.start();
30         t2.start();
31 
32         for( int i=0;i<10;i++ )
33         {
34             System.out.println(Thread.currentThread().getName()+"========"+i);
35             //让主线程临时停顿
36             Thread.yield();
37         }
38 
39         System.out.println("........................."+t);
40     }
41 }

4、join和yield方法介绍

  join()  让某个线程强制加入到当前cpu的执行队列中。在使用join加入其他线程到运行状态中来时,这个加入的语句肯定是写在其他的线程的执行代码中。书写加入线程的代码所在的那个线程此时就会处理临时冻结状态,等待被加入的线程执行结束后,它就立刻恢复运行的状态。

  yield()  让当前正在执行的临时停顿一下,然后立刻就恢复运行状态。

 

 多线程需要掌握的技术:

  1、创建线程的2种方式  继承Thread和实现Runnable接口

  2、掌握线程的安全问题发生的原因和解决

  3、掌握单生产和单消费的原理以及等待唤醒机制

5、面试题

  

  

posted @ 2015-11-22 21:50  灰太郎^_^  阅读(621)  评论(0编辑  收藏  举报