线程:是程序执行的一个路径,每个线程都有自己的局部变量表,程序计数器(指向正在执行的指令指针),以及各自的生命周期。、
创建一个线程
方法一
public class MyThread extends Thread{
public MyThread() {
}
@Override
public void run() {
for (int i = 0; i < 100; i++){
System.out.println(getName() + ":" + i);
}
}
}
public class MyThreadTest {
public static void main(String[] args) {
//创建线程对象
MyThread my = new MyThread();
//启动线程,run()相当于普通方法的调用,单线程效果
//my.run();
//首先启动了线程,然后再由jvm调用该线程的run()方法,多线程效果
my.start();
//两个线程演示,多线程效果需要创建多个对象而不是一个对象多次调用start()方法
MyThread my1 = new MyThread();
MyThread my2 = new MyThread();
my1.start();
my2.start();
}
}
方法二
public class Multi implements Runnable{
public void run() {
System.out.println("反手就是一个线程运行中~");
}
public static void main(String[] args) {
Multi m = new Multi();
Thread t1 = new Thread(m);
t1.start();
}
}
这里面的区别就是:如果没有继承Thread类,你创建的对象自然不是一个线程对象,所以你要明确创建一个线程类对象,来接收你实现的这个接口。
线程的生命周期
线程的生命周期包含5个阶段,包括:新建、就绪、运行、阻塞、销毁。
- 新建:就是刚使用new方法,new出来的线程;
- 就绪:就是调用的线程的start()方法后,这时候线程处于等待CPU分配资源阶段,谁先抢的CPU资源,谁开始执行;
- 运行:当就绪的线程被调度并获得CPU资源时,便进入运行状态,run方法定义了线程的操作和功能;
- 阻塞:在运行状态的时候,可能因为某些原因导致运行状态的线程变成了阻塞状态,比如sleep()、wait()之后线程就处于了阻塞状态,这个时候需要其他机制将处于阻塞状态的线程唤醒,比如调用notify或者notifyAll()方法。唤醒的线程不会立刻执行run方法,它们要再次等待CPU分配资源进入运行状态;
- 销毁:如果线程正常执行完毕后或线程被提前强制性的终止或出现异常导致结束,那么线程就要被销毁,释放资源;
完整的生命周期图如下:
新建状态
我们来看下面一段代码:
Thread t1 = new Thread();
这里的创建,仅仅是在JAVA的这种编程语言层面被创建,而在操作系统层面,真正的线程还没有被创建。只有当我们调用了 start() 方法之后,该线程才会被创建出来,进入Runnable状态。只有当我们调用了 start() 方法之后,该线程才会被创建出来
就绪状态
调用start()方法后,JVM 进程会去创建一个新的线程,而此线程不会马上被 CPU 调度运行,进入Running状态,这里会有一个中间状态,就是Runnable状态,你可以理解为等待被 CPU 调度的状态
t1.start()
用一张图表示如下:
那么处于Runnable状态的线程能发生哪些状态转变?
Runnable状态的线程无法直接进入Blocked状态和Terminated状态的。只有处在Running状态的线程,换句话说,只有获得CPU调度执行权的线程才有资格进入Blocked状态和Terminated状态,Runnable状态的线程要么能被转换成Running状态,要么被意外终止。
运行状态
当CPU调度发生,并从任务队列中选中了某个Runnable线程时,该线程会进入Running执行状态,并且开始调用run()方法中逻辑代码。
那么处于Running状态的线程能发生哪些状态转变?
- 被转换成Terminated状态,比如调用 stop() 方法;
- 被转换成Blocked状态,比如调用了sleep, wait 方法被加入 waitSet 中;
- 被转换成Blocked状态,如进行 IO 阻塞操作,如查询数据库进入阻塞状态;
- 被转换成Blocked状态,比如获取某个锁的释放,而被加入该锁的阻塞队列中;
- 该线程的时间片用完,CPU 再次调度,进入Runnable状态;
- 线程主动调用 yield 方法,让出 CPU 资源,进入Runnable状态
阻塞状态
Blocked状态的线程能够发生哪些状态改变?
- 被转换成Terminated状态,比如调用 stop() 方法,或者是 JVM 意外 Crash;
- 被转换成Runnable状态,阻塞时间结束,比如读取到了数据库的数据后;
- 完成了指定时间的休眠,进入到Runnable状态;
- 正在wait中的线程,被其他线程调用notify/notifyAll方法唤醒,进入到Runnable状态;
- 线程获取到了想要的锁资源,进入Runnable状态;
- 线程在阻塞状态下被打断,如其他线程调用了interrupt方法,进入到Runnable状态;
终止状态
一旦线程进入了Terminated状态,就意味着这个线程生命的终结,哪些情况下,线程会进入到Terminated状态呢?
- 线程正常运行结束,生命周期结束;
- 线程运行过程中出现意外错误;
- JVM 异常结束,所有的线程生命周期均被结束。