多线程作为Java中很重要的一个知识点,在此还是有必要总结一下的。
一.线程的生命周期及五种基本状态
关于Java中线程的生命周期,首先看一下下面这张较为经典的图:
上图中基本上囊括了Java中多线程各重要知识点。掌握了上图中的各知识点,Java中的多线程也就基本上掌握了。主要包括:
Java线程具有五中基本状态
新建状态(New):当线程对象对创建后,即进入了新建状态,如:Thread t = new MyThread();
就绪状态(Runnable):当调用线程对象的start()方法(t.start();),线程即进入就绪状态。处于就绪状态的线程,只是说明此线程已经做好了准备,随时等待CPU调度执行,并不是说执行了t.start()此线程立即就会执行;
运行状态(Running):当CPU开始调度处于就绪状态的线程时,此时线程才得以真正执行,即进入到运行状态。注:就 绪状态是进入到运行状态的唯一入口,也就是说,线程要想进入运行状态执行,首先必须处于就绪状态中;
阻塞状态(Blocked):处于运行状态中的线程由于某种原因,暂时放弃对CPU的使用权,停止执行,此时进入阻塞状态,直到其进入到就绪状态,才 有机会再次被CPU调用以进入到运行状态。根据阻塞产生的原因不同,阻塞状态又可以分为三种:
1.等待阻塞:运行状态中的线程执行wait()方法,使本线程进入到等待阻塞状态;
2.同步阻塞 -- 线程在获取synchronized同步锁失败(因为锁被其它线程所占用),它会进入同步阻塞状态;
3.其他阻塞 -- 通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。
死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。
二. Java多线程的创建及启动
【通过继承Thread】
一个Thread对象只能创建一个线程,即使它调用多次的.start()也会只运行一个的线程。
【看下面的代码 & 输出结果】
class CTest extends Thread { 4 private int tickte = 20; 5 6 public void run() { 7 while (true) { 8 if (tickte > 0) { 9 System.out.println(Thread.currentThread().getName() + " 出售票 "10 + tickte--);11 } else {12 System.exit(0);13 }14 }15 }16 17 }18 19 public class Demo3 {20 public static void main(String[] args) {21 // new CTest().start();22 // new CTest().start();23 Thread t1 = new CTest();//创建一个线程24 t1.start();25 t1.start();26 }27 }28 29 //30 Thread-0 出售票 2031 Thread-0 出售票 1932 Thread-0 出售票 1833 Thread-0 出售票 1734 Thread-0 出售票 1635 Thread-0 出售票 1536 Thread-0 出售票 1437 Thread-0 出售票 1338 Thread-0 出售票 1239 Thread-0 出售票 1140 Thread-0 出售票 1041 Thread-0 出售票 942 Thread-0 出售票 843 Thread-0 出售票 744 Thread-0 出售票 645 Thread-0 出售票 546 Thread-0 出售票 447 Thread-0 出售票 348 Thread-0 出售票 249 Thread-0 出售票 1
通过调用当前线程对象的名字Thread.currentThread.getName(),根据结果可以看出,只运行了一个线程。
这就说明了一个问题,每创建一个Thread对象,只能创建一个线程。
2.实现Runnable接口,并重写该接口的run()方法,该run()方法同样是线程执行体,创建Runnable实现类的实例,并以此实例作为Thread类的target来创建Thread对象,该Thread对象才是真正的线程对象。
class MyRunnable implements Runnable { 2 private int i = 0; 3 4 @Override 5 public void run() { 6 for (i = 0; i < 100; i++) { 7 System.out.println(Thread.currentThread().getName() + " " + i); 8 } 9 }10 }
public class ThreadTest { 2 3 public static void main(String[] args) { 4 for (int i = 0; i < 100; i++) { 5 System.out.println(Thread.currentThread().getName() + " " + i); 6 if (i == 30) { 7 Runnable myRunnable = new MyRunnable(); // 创建一个Runnable实现类的对象 8 Thread thread1 = new Thread(myRunnable); // 将myRunnable作为Thread target创建新的线程 9 Thread thread2 = new Thread(myRunnable);10 thread1.start(); // 调用start()方法使得线程进入就绪状态11 thread2.start();12 }13 }14 }15 }
3、使用匿名内部类的方式创建线程
首先回顾下之前的匿名内部类:
匿名内部类的格式:
new 接口或者接口名(){ 重写方法 };本质:是该类或者接口的子类对象
匿名内部类方式使用多线程
1、new Thread(){代码…}.start();2、new Thread(new Runnable(){代码…}).start();例子1:继承Thread类的匿名内部类实现多线程
// 一、继承Thread类实现多线程2 new Thread() {3 // 线程的代码4 public void run() {5 for (int x = 0; x < 100; x++) {6 System.out.println("Thread" + "--" + x);7 }8 }9 }.start();// 别忘了启动线程
例子2:继承Runnable类的匿名内部类实现多线程
// 二、继承Runnable类实现多线程 2 new Thread(new Runnable() { 3 // 线程的代码 4 public void run() { 5 for (int x = 0; x < 100; x++) { 6 System.out.println("Runnable" + "--" + x); 7 } 8 } 9 10 })11 12 {13 // 这里的代码为空14 }.start();
由于继承Runnable类实现线程中,start之前的{}为空,这里在继承Thread类中是重写线程的方法的,
所以,如果两者结合起来的话,会执行Runnable还是Thread?
例子3:同时继承Runnable类和Thread类的匿名内部类来实现多线程
// 三、两者结合 2 3 new Thread(new Runnable() { 4 5 public void run() { 6 // 填写继承Ruannble的线程代码 7 for (int x = 0; x < 100; x++) { 8 System.out.println("hello" + "--" + x); 9 }10 11 }12 13 }) {14 // 填写继承Thread类的线程代码15 public void run() {16 for (int x = 0; x < 100; x++) {17 System.out.println("world" + "--" + x);18 }19 }20 }.start();21 //通过运行结果可知道,这里只执行继承Thread类的代码