Eason-S

导航

java线程小结1

 1.创建线程的两种方法

  新线程的创建和启动都是通过java代码触发的。除了第一个线程(也就是启动程序的。运行main()方法的线程)是由java平台直接创建的之外,其余的线程都是在java代码中通过“创建Thread类的实例,启动线程”这种方式创建并启动的。

  当启动一个新的线程时,其过程是:由java代码通知java平台,java平台再启动线程。线程1启动线程2的过程实际就是线程1执行thread.start(),这对于线程1来说是一个很短的过程,因为启动线程的具体工作都是java平台做的,线程1只是通知而已,通知完了就继续执行代码,不等待线程2。

 a.继承java.lang.Thread类

  使用一个类去继承Thread类,然后为这个Threa类的子类添加一个run()方法,用来覆盖Thread类中原来的run()方法。

  使用继承自Thread类的子类的方法创建线程时,不需要向该子类中传入参数。如:MyThread thread = new MyThread();  //其中MyThread类是继承自Thread类的子类

 b.实现java.lang.Runnable接口

  使用一个实现了Runnable接口的类,在该类的内部有run()方法。

  使用实现了Runnable接口的类的方法创建线程时,需要向Thread中传入该类的实例。如:Thread thread = new Thread(MyRunnable);  //其中MyRunnable实现了Runnable接口的类的实例

 c.两种方式的比较

  实际中往往采用实现Runable接口,一方面因为java只支持单继承,继承了Thread类就无法再继续继承其它类,而且Runable接口只有一个run方法。

 

 2.start和run两种方法的区别

  run()方法:在本线程内调用该Runnable对象的run()方法,可以重复多次调用;
  start()方法:启动一个线程,调用该Runnable对象的run()方法,不能多次启动一个线程;

 a.start:

  用start方法来启动线程,真正实现了多线程运行,这时无需等待run方法体代码执行完毕而直接继续执行下面的代码。通过调用Thread类的start()方法来启动一个线程,这时此线程处于就绪(可运行)状态,并没有运行,一旦得到cpu时间片,就开始执行run()方法,这里方法 run()称为线程体,它包含了要执行的这个线程的内容,Run方法运行结束,此线程随即终止。

 b.run:

  run()方法只是类的一个普通方法而已,如果直接调用Run方法,程序中依然只有主线程这一个线程,其程序执行路径还是只有一条,还是要顺序执行,还是要等待run方法体执行完毕后才可继续执行下面的代码,这样就没有达到写线程的目的。

  run()方法中编写需要在独立线程内执行的代码,run()方法可以调用其他方法,但是执行的线程总是通过调用run()方法开始的。没有参数的run()方法是自动被调用的,而带参数的run()方法是被重载的,必须显式调用。

  总结:调用start方法方可启动线程,而run方法只是thread的一个普通方法调用,还是在主线程里执行。这两个方法应该都比较熟悉,把需要并行处理的代码放在run()方法中,start()方法启动线程将自动调用 run()方法,这是由jvm的内存机制规定的。并且run()方法必须是public访问权限,返回值类型为void。

 

 3.实例

 1 package com.ljq.test;
 2 
 3 public class ThreadTest {
 4     
 5     /**
 6      * 观察直接调用run()和用start()启动一个线程的差别 
 7      * 
 8      * @param args
 9      * @throws Exception
10      */
11     public static void main(String[] args){
12         Thread thread=new ThreadDemo();
13         //第一种
14         //表明: run()和其他方法的调用没任何不同,main方法按顺序执行了它,并打印出最后一句
15         //thread.run();
16         
17         //第二种
18         //表明: start()方法重新创建了一个线程,在main方法执行结束后,由于start()方法创建的线程没有运行结束,
19         //因此主线程未能退出,直到线程thread也执行完毕.这里要注意,默认创建的线程是用户线程(非守护线程)
20         //thread.start();
21         
22         //第三种
23         //1、为什么没有打印出100句呢?因为我们将thread线程设置为了daemon(守护)线程,程序中只有守护线程存在的时候,是可以退出的,所以只打印了七句便退出了
24         //2、当java虚拟机中有守护线程在运行的时候,java虚拟机会关闭。当所有常规线程运行完毕以后,
25         //守护线程不管运行到哪里,虚拟机都会退出运行。所以你的守护线程最好不要写一些会影响程序的业务逻辑。否则无法预料程序到底会出现什么问题
26         //thread.setDaemon(true);
27         //thread.start();
28         
29         //第四种
30         //用户线程可以被System.exit(0)强制kill掉,所以也只打印出七句
31         thread.start();
32         System.out.println("main thread is over");
33         System.exit(1);
34     }
35     
36     public static class ThreadDemo extends Thread{
37         @Override
38         public void run() {
39             for (int i = 0; i < 100; i++) {
40                 System.out.println("This is a Thread test"+i);
41             }
42         }
43     }
44 }

 

  

posted on 2016-05-20 23:02  Eason_S  阅读(305)  评论(0编辑  收藏  举报