申明:
内容来源于老师的课堂笔记,这只是我的整理以及一些额外补充。
十一、线程
了解点
- 线程的概念
-
线程并发:多个线程同时运行,其实线程并发只是一种表象,实际上是因为cpu的运算速度快而让使用者感觉不到等待。Cpu接到线程执行指令时是抢占式调度执行的
-
对于Java应用程序而言,有三个基础线程:main线程(主线程)、GC线程、异常线程
-
- 线程优先级
-
- JUC
- 概念
- 实现
- 创建两个等待线程
- 创建一个或多个唤醒线程
- 测试
- 创建两个等待线程
- 作用
- 概念
掌握点
- 线程的开发
- 继承Thread类,重写run()
- run()是由JVM调度执行的,线程中不能手动调用(手动调用无线程效果)
- eg:
public class Thread1 extends Thread{ public void run() { System.out.println("----------Thread1-run()------------"); } } public static void main(String[] args) { Thread t1 = new Thread1(); t1.start(); }
- 实现Runnable接口,重写run()
- eg:
public class Thread2 implements Runnable{ public void run() { System.out.println("----------Thread2-run()------------"); } } public class Test { public static void main(String[] args) { Runnable tt2 = new Thread2(); Thread t2 = new Thread(tt2); t2.start(); } }
- eg:
- 两种创建方式的比较
- 继承Thread类,重写run()
- 线程的方法
- 基础方法
-
public class Thread1 extends Thread{ public void run() { this.setName("1"); this.getName(); this.isAlive(); System.out.println("----------Thread1-run()------------"); } }
-
- 线程的基本状态
- Thread类中有一个枚举类State,记录了线程的生命周期各种状态
- 线程的阻塞-sleep()
- 运行中的线程由于外部资源的缺失或者调用了sleep方法,则会进入阻塞状态
- sleep()方法让当前线程进入睡眠/阻塞状态,等待指定时间后,线程自动醒来恢复到可运行状态。
- 一个对象调用sleep()后,不会释放当前对象的状态(对象所持有的对象锁),所以不会影响其他线程的运行。
- sleep()指定的睡眠时间无法精确的控制线程的运行流程
- Join()
- join()方法同样可以导致当前线程进入阻塞状态
- 执行特点:当前线程中如果调用了另外一个线程的join方法,当前线程会立即阻塞,直到另一个线程完全执行完成。
- yield()
- yield()方法不会进入到阻塞状态,当线程调用此方法后,当前线程释放掉系统资源,由系统再次调用执行,此时,当前线程继续抢占资源,有可能继续抢占到资源。
- 基础方法
- 线程练习
- 实现方法:
- 栈的大小固定,先使用普通方式实现栈
- 在①中把pop和push通过两个线程实现
- 两个线程操作的是同一个数组
- 运行下栈的操作,看发现什么问题?
- 实现方法:
- 线程安全
- 线程不安全:多个线程并发访问了同一个临界资源对象,若破坏了对象的原子操作,从而有可能造成数据不一致。
- 要解决线程安全问题,就要保障原子操作不被破坏
- 解决线程安全问题的方法:
- 用synchronized修饰语句块(原子操作)
- 用synchronized修饰方法
- 注意
- synchronized关键字可以修饰run()方法同样能够实现线程同步
- 面试题:
-
多线程操作同类不同对象中的静态方法[Synchronized修饰]时,会不会同样能够实现线程同步?按正常理解多个线程实现对同一个对象的操作,此时验证多线程对多个对象操作,看static方法能否可以线程同步?
-
- Lock锁定线程
- 可以通过Lock接口对象去锁定原子操作,以达到线程同步的目的
-
通过try...finally方法确保锁一定被释放
-
面试题
-
练习
- 解决线程安全问题的方法:
- 线程通信
- 死锁
- 当两个线程彼此占用对方所需要的对象锁,且彼此都不释放对象锁资源时,即两个线程进入死锁状态。
- 可以通过Object中的wait()和notify()方法来实现线程之间的通信【这两个方法一定是在synchronized语句中执行的】
- 线程通信实现
- 生产者与消费者问题
- 两个线程操作同一个对象:一个线程只负责不断地生产资源,另一个线程只负责不断地消费资源
- 生产者与消费者问题
- 死锁
- Java阻塞队列
- 阻塞队列
- 队列-Queue-FIFO
- ArrayBlockingQueue
- 类结构图
- API值对
- 实现
- 基本API:
- ArrayBlockingQueue源码分析
- 基本属性定义
- Take()
- Put(e)
- 总结
- 阻塞队列
- 单例模式的线程安全问题
- Callable接口创建线程
- Callable说明
- Callable是Java5中新增的创建线程的方法
- Callable是一个普通的接口,需要借助FutureTask类来实现线程的功能,FutureTask是实现了Runnable接口的,所以线程真正执行的还是run(),也需要借助Thread类中的start()启动线程。
- Callable使用
- Callable说明
- 线程池
- 概念
- 线程池涉及的接口和类:
- 实现
- 创建一个Thread1、2两个线程,交给ExecutorService进行管理
- 管理线程池
- 概念
- 线程类型
- 线程面试题
- 如何让所有线程都执行完后再打印一条日志记录(即让主线程最后执行完)