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 }