一、概念
1、进程
2、线程
- 是1个进程(程序内部)的1条执行路径
- 单线程:1个进程中只有1个线程(1条执行路径)
- 多线程:1个进程中包含多个线程(多条执行路径)
- 线程之间堆内存、方法区内存共享;但是栈内存独立,1个线程一个栈
3、CPU与线程的关系
- 单核CPU:不能够做到真正的多线程并发,因为在一个时间单元内,只能执行一个线程的任务,多个线程谁获取时间片运行谁,每个线程获取时间片的概率相等,可能:t1、t2、t2、t2;给人一种多线程并发的感觉:其实是由于CPU的处理速度极快,多个线程之间频繁切换执行,跟人来的感觉是多个事情同时在做
4、并行、并发
- 并行:多核CPU同时执行多个任务。比如:多个人同时做不同的事
- 并发:单核CPU同时执行多个任务。比如:多个人做同一件事
5、线程的生命周期
- 新建状态
- 就绪状态
- 线程调用了start方法等待获取CPU时间片
- 表示当前线程具有抢夺CPU时间片的权力
- 运行状态
- 线程对象开始执行run方法
- run方法的开始执行标志着这个线程进入运行状态,当之前占有的CPU时间片用完之后,会重新回到就绪状态继续抢夺CPU时间片,当再次抢到CPU时间之后,会重新进入run方法接着上一次的代码继续往下执行
- 阻塞状态
- 当一个线程遇到阻塞事件,例如:sleep方法、获取synchronized排他锁失败(因为锁被其它线程所占用)等,此时线程会进入阻塞状态,阻塞状态的线程会放弃之前占有的CPU时间片,之前的时间片没了需要再次回到就绪状态抢夺CPU时间片
- 死亡状态
- run方法执行完毕或者因异常退出了run方法,该线程生命周期结束
二、创建
1、继承Thread
class Thread implements Runnable
- 缺点:
- 由于java是单继承的,这导致继承了Thread后就不能在继承其它类了;在实际开发中会经常继承某个超类来复用其中的方法,这导致两者不能同时继承
- 继承线程后重写run方法来定义任务,这又导致我们将任务直接定义在线程上使得线程只能做该任务,无法并发执行其他任务,重用性变差
public class HandleMsg extends Thread{
/** 线程标识 */
private String threadKey;
/** 构造方法用来区分不同线程便于测试 */
public HandleMsg(String threadKey){
this.threadKey=threadKey;
}
@Override
public void run(){
for(int i=0;i<10;i++){
System.err.println(threadKey+":run");
}
}
}
//匿名内部类lambda
Thread t1=new Thread(()->{
for(int i=0;i<10;i++) {
System.err.println("t1:run");
}
};
//创建2个线程对象
HandleMsg h1=new HandleMsg("h1");
HandleMsg h2=new HandleMsg("h2");
//启动线程,开始执行实现的run方法
h1.start();
h2.start();
2、实现Runnable接口
public class HandleMsg implements Runnable{
/** 线程标识 */
private String threadKey;
/** 构造方法用来区分不同线程便于测试 */
public HandleMsg(String threadKey){
this.threadKey=threadKey;
}
@Override
public void run(){
for(int i=0;i<10;i++){
System.err.println(threadKey+":run");
}
}
}
//创建2个任务对象
HandleMsg h1=new HandleMsg("h1");
HandleMsg h2=new HandleMsg("h2");
//将任务1交给线程1
Thread t1=new Thread(h1);
//将任务2交给线程2
Thread t2=new Thread(h2);
//启动线程,开始执行任务的run方法
t1.start();
t2.start();
3、实现Callable接口
FutureTask implements RunnableFuture
,RunnableFuture<V> extends Runnable
- 优点
- 线程和线程执行的任务分离
- 有返回值
- 可以声明抛出的异常
//Callable的泛型就是重写的call方法的返回值类型
public class HandleMsg implements Callable<String>{
/** 线程标识 */
private String threadKey;
/** 构造方法用来区分不同线程便于测试 */
public HandleMsg(String threadKey){
this.threadKey=threadKey;
}
@Override
public String call() throws Exception{
for(int i=0;i<10;i++){
System.err.println(threadKey+":run");
}
return threadKey;
}
}
//创建2个任务对象
HandleMsg h1=new HandleMsg("h1");
HandleMsg h2=new HandleMsg("h2");
//创建FutureTask类包装任务对象,泛型就是任务类实现Callable的泛型,也就是call方法返回值的类型
FutureTask<String> f1=new FutureTask<>(h1);
FutureTask<String> f2=new FutureTask<>(h2);
//将FutureTask对象交给线程
Thread t1=new Thread(f1);
Thread t2=new Thread(f2);
//启动线程,开始执行任务的run方法
t1.start();
t2.start();
//获取返回值
String result1=f1.get();
String result2=f2.get();
三、API
1、获取运行使用的线程
//在哪个方法执行就获取执行该方法的线程
Thread thread=Thread.currentThread();
2、唯一标识
long getId();
3、线程名
String getName();
//线程启动之前可以设置线程名
void setName(String name);
4、优先级
int getPriority();
//启动之前设置
//线程无法主动获取cpu时间片,唯一可以干涉线程调度工作的方式就是修改线程的优先级,最大程度的改善获取cpu时间片的几率,理论上,线程优先级越高的线程获取cpu时间片的次数越多
void setPriority(int newPriority);
5、是否处于活动状态
boolean isAlive();
6、守护线程
- 守护线程又称为后台线程,默认创建出来的线程都是普通线程或称为前台线程
- 当进程结束时,所有正在运行的守护线程都会被强制中断
- 进程的结束:当一个进程中没有任何前台线程时即结束
- main主线程就是前台线程,不受其他线程影响,分配其他线程后接着干自己的事,其他线程执行的时候,main线程可能已经结束了
//是否为守护线程
boolean isDaemon();
//启动之前设置
void setDaemon(boolean on);
7、join
- 作用是:让当前执行的线程陷入等待(内部调用了wait方法)。
- 永久等待:其实现原理是不停的检查当前线程是否存活,该线程的任务执行完毕后就会处于死亡状态,如果存活则说明任务还未执行完毕-继续等待
- 线程启动之后调用
1、API
//等待线程执行完任务后再往下执行,join(0);
void join();
//等待线程执行一段时间后再往下执行,无论线程任务是否执行完毕,单位:毫秒
void join(long millis);
2、有无join对比
//1、程序正常运行顺序是不会等待线程任务执行完毕,就会往下执行
Thread t1=new Thread(()->{
for(int i=0;i<5;i++){
System.err.println(i);
}
});
t1.start();
System.err.println("next task");
// 输出结果
// next task
// 0
// 1
// 2
// 3
// 4
//2、调用join
Thread t1=new Thread(()->{
for(int i=0;i<5;i++){
System.err.println(i);
}
});
t1.start();
t1.join();
System.err.println("next task");
// 0
// 1
// 2
// 3
// 4
// next task
8、yield
- 线程让步
- 暂停(不是终止)当前正在执行的线程任务,让其它具有相同优先级或更高优先级的等待的线程执行任务(其它也会包含暂停的线程,所以有可能刚暂停就执行)
- 暂停的线程状态变化:运行状态 -> 就绪状态
- 暂停期间不会释放锁,所以其他线程获取不到锁
9、sleep
- 使线程睡眠
- sleep的睡眠期间不会释放锁,所以其它线程获取不到锁
- 睡眠线程的状态变化:运行状态 -> 阻塞状态 -> 就绪状态
//Thread的静态方法,单位:毫秒
static void sleep(long millis) throws InterruptedException;
10、线程中断
//线程是否中断
boolean isInterrupted();
//中断线程睡眠
Thread t1=new Thread(()->{
System.err.println("run......");
try{
//睡眠10s
Thread.sleep(10000L);
}catch(InterruptedException e){
e.printStackTrace();
}
System.err.println("end......");
});
//启动线程
t1.start();
//中断睡眠
//线程不会睡眠10s,而是被中断,抛出InterruptedException,捕获该异常后,继续执行后续的代码
t1.interrupt();
System.err.println("next task");
// 执行结果
// next task
// run......
// end......