Java多线程总结

来源:互联网 发布:免费域名官网 编辑:程序博客网 时间:2024/06/05 15:47
1、多线程有哪几种实现方法?举个例子说明下线程的同步。

(1)Java多线程有两种实现方式:继承Thread类和实现Runnable接口,Thread就是实现了Runnable接口。

两个最简单的线程例子:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. package chc.runnable;  
  2.   
  3. public class ThreadTest2 {  
  4.     public static void main(String[] args) throws InterruptedException {  
  5.         Thread1 t = new Thread1();  
  6.         //t.run(); //这里也不能直接调用方法  
  7.         t.start();  
  8.         for (int i = 0; i < 100; i++) {  
  9.             System.out.println("main:"+i);  
  10.         }  
  11.     }  
  12. }  
  13.   
  14. class Thread1 extends Thread{  
  15.     public void run() {  
  16.         for (int i = 0; i < 100; i++) {  
  17.             System.out.println("Thread-----:"+i);  
  18.         }  
  19.     }  
  20. }  
[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. package chc.runnable;  
  2.   
  3. public class ThreadTest1 {  
  4.     public static void main(String[] args) {  
  5.         Runnable1 r =new Runnable1();  
  6.         Thread t1=new Thread(r);  
  7.         t1.start();  
  8.     }  
  9. }  
  10.   
  11. class Runnable1  implements Runnable {  
  12.   
  13.     public void run() {  
  14.         // TODO Auto-generated method stub  
  15.         for (int i = 1; i <= 5; i++) {  
  16.             System.out.println("实现Runnable接口的线程----->"+i);  
  17.         }  
  18.     }  
  19.   
  20. }  

通过上两个例子发现,当启动线程的时候并不影响主程序的继续执行。

(2)线程同步问题

使用synchronnized关键字

银行账户存取钱的问题:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. public class ThreadTest {  
  2.     public static void main(String[] args){  
  3.         ThreadA  t1=new ThreadA();  
  4.         t1.start();  
  5.         ThreadB  t2=new ThreadB();  
  6.         t2.start();  
  7.     }  
  8. }  
  9.   
  10. class ThreadA extends Thread{  
  11.     @Override  
  12.     public void run(){  
  13.         for(int i=0;i<100;i++){  
  14.             account.add();  
  15.             try {  
  16.                 sleep(10);//模拟银行系统处理时间  
  17.             } catch (InterruptedException e) {  
  18.                 // TODO Auto-generated catch block  
  19.                 e.printStackTrace();  
  20.             }  
  21.         }  
  22.     }  
  23. }  
  24.   
  25. class ThreadB extends Thread{  
  26.     @Override  
  27.     public void run() {  
  28.         for(int i=0;i<100;i++){  
  29.             account.remove();  
  30.             try {  
  31.                 sleep(10);<span style="font-family: SimSun;">//模拟银行系统处理时间</span>  
  32.             } catch (InterruptedException e) {  
  33.                 // TODO Auto-generated catch block  
  34.                 e.printStackTrace();  
  35.             }  
  36.         }  
  37.     }  
  38. }  
  39.   
  40. class account {  
  41.     public static   int count=1000;  
  42.       
  43.     //减去100元  
  44.     public static synchronized void remove(){  
  45.         count=count-100;  
  46.         System.out.println("减去100元,卡内余额"+count);  
  47.     }  
  48.       
  49.     //增加100元  
  50.     public static synchronized void add(){  
  51.         count=count+100;  
  52.         System.out.println("加上100元,卡内余额"+count);  
  53.     }  
  54. }  
注意:上例中将account类的add()和remove()方法都加上了static关键字,因为这样在线程中可以直接通过类名调用到这两个方法,而不需要实例化对象。

因为synchronized同步只对同一个对象中的方法有效,也就是说一个线程正在执行account的add()方法,另一个线程是可以调用到另一个对象的add方法的。

2、启动一个线程是用run()还是start(),调用的时候有什么区别?

当然是start()了,当调用线程的start()方法的时候,线程就会进入到就绪状态

run()方法是线程的执行入口,当线程从就绪状态进入到执行状态时首先要从run()方法开始执行。

当然,我们也是可以直接通过线程对象调用该对象的run()方法的,只是这只是一次普通的调用,并没有启动任何一个线程。

当我们调用start()方法时,是另外启动了一个线程去执行线程类的代码,并不影响主程序的执行,但是调用run()方法的时候要等待run()方法内的代码执

行完主程序才可以向下执行,举个例子:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. public class ThreadDemo2 {  
  2.     public static void main(String[] args) {  
  3.         Thread5 t1=new Thread5();  
  4.         Thread6 t2=new Thread6();  
  5.         t1.run();  
  6.         t2.start();  
  7.         for(int i=0;i<100;i++){  
  8.             System.out.println("主进程执行:"+i);  
  9.         }  
  10.     }  
  11. }  
  12.   
  13. class Thread5 extends Thread{  
  14.     @Override  
  15.     public void run() {  
  16.         for(int i=0;i<100;i++){  
  17.             System.out.println("Thread5执行:"+i);  
  18.         }  
  19.     }  
  20. }  
  21.   
  22. class Thread6 extends Thread{  
  23.     @Override  
  24.     public void run() {  
  25.         for(int i=0;i<100;i++){  
  26.             System.out.println("Thread6执行:"+i);  
  27.         }  
  28.     }  
  29. }  
输出结果的顺序是:Thread5全部打印完后 Thread6和主程序交替打印。验证了上面的说法

3、当一个线程进入到一个对象的synchronized方法,那么其他线程是否可以进入该对象的其它方法?

不一定,看情况

如果其它方法加了static关键字,那么该方法属于类,不属于对象,不能与对象的方法保持同步(即使有synchronized关键字),是能进入的。

如果其它方法不带有static关键字且带有synchronized关键字,那么不能进入,如果不带,则能。

再其次就看方法内部有没有wait()方法释放锁了

4、子线程循环2次,接着主线程循环3次,接着子线程循环3次,接着主线程循环3次,如此循环5次,请写出程序代码。

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. public class ThreadDemo5 {  
  2.     static boolean thread_flag=false;//指示子线程循环是否结束  
  3.     static boolean main_flag=true;   //调用子线程的start方法后变为true,循环等待thread_flag为true(也就是子线程循环完)的时刻,主线程循环完又变为false;  
  4.     public static void main(String[] args) {  
  5.         for(int k=0;k<5;k++){  
  6.             Thread7 t=new Thread7();  
  7.             t.start();  
  8.             main_flag=true;  
  9.             while (main_flag) {//循环等待thread_f  
  10.                 if(thread_flag){  
  11.                     for(int i=0;i<3;i++){  
  12.                         System.out.println("主线程第一次循环"+(i+1)+"次");  
  13.                     }  
  14.                     thread_flag=false;  
  15.                     main_flag=false;  
  16.                 }     
  17.             }  
  18.         }  
  19.     }  
  20. }  
  21.   
  22. class Thread7 extends Thread {  
  23.     static boolean flag=true;//标志子线程第几次循环,当值为true的时候子线程循环2次,否则循环3次  
  24.     public void run() {  
  25.         if(flag){  
  26.             for(int i=0;i<2;i++){  
  27.                 System.out.println("子线程第一次循环"+(i+1)+"次");  
  28.             }  
  29.             flag=false;  
  30.         }else{  
  31.             for(int i=0;i<3;i++){  
  32.                 System.out.println("子线程第二次循环"+(i+1)+"次");  
  33.             }  
  34.             flag=true;  
  35.         }  
  36.         ThreadDemo5.thread_flag=true;  
  37.           
  38.           
  39.     }  
  40. }  

这个题就是要注意调用子线程的start方法的时候并不能阻止主程序继续向下执行,所以我们要用变量来标记。

5、sleep()和wait()有何异同?

(1)首先一个最明显的区别是  wait是Object类的方法,而sleep()是Thread类的静态方法,谁调用了该方法谁去休眠,即使在a线程里调用了b线程的sleep方法,实际上还是a线程去休眠.

(2)比较重要的一点是sleep没有释放出锁,而wait释放了锁,是其他线程可以使用同步块资源。

     sleep不出让系统资源;wait是进入线程等待池等待,出让系统资源,其他线程可以占用CPU。一般wait不会加时间限制,因为如果wait线程的运行资源不够,再出来也没用,要

     等待其他线程调用notify/notifyAll唤醒等待池中的所有线程,才会进入就绪队列等待OS分配系统资源。sleep(milliseconds)可以用时间指定使它自动唤醒过来,如果时间不到

     只能调用interrupt()强行打断。

(3)使用范围:

     wait,notify和notifyAll只能在同步控制方法或者同步控制块里面使用,而sleep可以在任何地方使用 
     synchronized(x){ 
       x.notify() 
       //或者wait() 
      }

(4)sleep需要捕获异常,而wait不需要。

6、现在有T1 T2 T3三个线程,怎样保证T2在T1执行完之后执行 T3在T2执行完之后执行

这题主要是考察对join()方法的使用。

当线程A当中执行了线程B.join(),那么A线程要等待B线程执行完才可以执行。

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. public class JoinDemo {  
  2.     public static void main(String[] args) {  
  3.         T1 t1=new T1("T1");  
  4.         T2 t2=new T2("T2");  
  5.         T3 t3=new T3("T3");  
  6.         t1.start();  
  7.         try {  
  8.             t1.join();  
  9.         } catch (InterruptedException e) {  
  10.             // TODO Auto-generated catch block  
  11.             e.printStackTrace();  
  12.         }  
  13.         t2.start();  
  14.         try {  
  15.             t2.join();  
  16.         } catch (InterruptedException e) {  
  17.             // TODO Auto-generated catch block  
  18.             e.printStackTrace();  
  19.         }  
  20.         t3.start();  
  21.     }  
  22. }  
  23.   
  24. class T1 extends  Thread{  
  25.     private String name;  
  26.     public T1(String name) {  
  27.         this.name=name;  
  28.     }  
  29.   
  30.     @Override  
  31.     public void run() {  
  32.         for(int i=0;i<5;i++){  
  33.             try {  
  34.                 sleep(5);  
  35.             } catch (InterruptedException e) {  
  36.                 // TODO Auto-generated catch block  
  37.                 e.printStackTrace();  
  38.             }  
  39.             System.out.println(this.name+"循环"+i);  
  40.         }  
  41.     }  
  42. }  
  43. class T2 extends  Thread{  
  44.     private String name;  
  45.     public T2(String name) {  
  46.         this.name=name;  
  47.     }  
  48.     @Override  
  49.     public void run() {  
  50.         for(int i=0;i<5;i++){  
  51.             try {  
  52.                 sleep(5);  
  53.             } catch (InterruptedException e) {  
  54.                 // TODO Auto-generated catch block  
  55.                 e.printStackTrace();  
  56.             }  
  57.             System.out.println(this.name+"循环"+i);  
  58.         }  
  59.     }  
  60. }  
  61. class T3 extends  Thread{  
  62.     private String name;  
  63.     public T3(String name) {  
  64.         this.name=name;  
  65.     }  
  66.     @Override  
  67.     public void run() {  
  68.         for(int i=0;i<5;i++){  
  69.             System.out.println(this.name+"循环"+i);  
  70.         }  
  71.     }  
  72. }  

7、简述synchronized和java.util.concurrent.locks.Lock的异同?

主要相同点:Lock能完成synchronized所实现的所有功能

主要不同点:Lock有比synchronized更精确的线程语义和更好的性能。synchronized会自动释放锁,而Lock一定要求程序员手工释放,并且必须在finally从句中释放。

Lock还有更强大的功能,例如,它的tryLock方法可以非阻塞方式去拿锁。

0 0