多线程

基本概念:程序、进程、线程

  • 程序(program)是为完成特定任务、用某种语言编写的一组指令的有序集合。即指一段静态的代码,其本身没有任何运行的含义,是一个静态对象。
  • 进程(process)是程序的一次执行过程,或是正在运行的一个程序。是一个动态的过程:有它自身的产生、存在和消亡的过程。——生命周期
    • 程序是静态的,进程是动态的
    • 进程作为系统资源分配的单位,系统在运行时会为每个进程分配不同的内存区域
  • 线程(thread),进程可进一步细化为线程,是一个程序内部的一条执行路径。
    • 若一个进程同一时间并行执行多个线程,就是支持多线程的
    • 线程是CPU调度和执行的单位,每个线程拥有独立的运行栈和程序计数器(pc),线程切换的开销小。
    • 通常一个进程中可以包含多个线程,当然一个进程中至少有一个线程,不然没有存在的意义。
    • 一个进程中的多个线程共享相同的内存单元/内存地址空间 —> 它们从同一堆中分配对象,可以访问相同的变量和对象,使得线程间通信更简便、高效。但多个线程操作共享的系统资源可能就会带来安全的隐患。

      注意:很多多线程是模拟出来的,真正的多线程是指有多个CPU,即多核,如服务器。如果是模拟出来的多线程,即在一个CPU的情况下,在同一个时间点,CPU只能执行一个代码,因为切换的很快,所以就有同时执行的错觉。

1)进程与线程

多线程 - 图1

  • 单核CPU和多核CPU的理解
    • 单核CPU,其实是一种假的多线程,因为在一个时间单元内,只能执行一个线程的任务。但是因为CPU时间单元特别短,因此感觉不出来。
    • 如果是多核的话,才能更好的发挥多线程的效率。
    • 一个Java应用程序java.exe,其实至少有三个线程:main()主线程,gc()垃圾回收线程,异常处理线程。当然如果发生异常,会影响主线程。
  • 并行与并发
    • 并行:多个CPU同时执行多个任务。比如:多个人同时做不同的事。
    • 并发:一个CPU(采用时间片)同时执行多个任务。比如:秒杀、多个人做同一件事。

2)普通方法调用和多线程

多线程 - 图2

3)使用多线程的优点

  • 提高应用程序的响应。对图形化界面更有意义,可增强用户体验。
  • 提高计算机系统CPU的利用率
  • 改善程序结构。将既长又复杂的进程分为多个线程,独立运行,利于理解和修改

4)何时需要多线程

  • 程序需要同时执行两个或多个任务。
  • 程序需要实现一些需要等待的任务时,如用户输入、文件读写操作、网络操作、搜索等。
  • 需要一些后台运行的程序时。

5)本章核心概念

  • 线程就是独立的执行路径;
  • 在程序运行时,即使没有自己创建线程,后台也会有多个线程,如主线程、gc线程;
  • main() 称之为主线程,为系统的入口,用于执行整个程序;
  • 在一个进程中,若开辟了多个线程,线程的运行由调度器(CPU)调度,调度器与操作系统紧密相关,先后顺序不能人为干预。
  • 对同一份资源操作时,会存在资源抢夺的问题,需要加入并发控制;
  • 线程会带来额外的开销,如cpu调度时间、并发控制开销。
  • 每个线程在自己的工作内存交互,内存控制不当会造成数据不一致。

线程创建和使用(重点)

  • Java语言的JVM允许程序运行多个线程,它通过java.lang.Thread类来体现。
  • Thread类的特性
    • 每个线程都是通过某个特定Thread对象的run()方法来完成操作的,经常把run()方法的主体称为线程体
    • 通过该Thread对象的start()方法来启动这个线程,而非直接调用run()
  • 构造器
    • Thread():创建新的Thread对象
    • Thread(String threadname):创建线程并指定线程实例名
    • Thread(Runnabletarget):指定创建线程的目标对象,它实现了Runnable接口中的run方法
    • Thread(Runnable target, String name):创建新的Thread对象

1)三种创建方式

多线程 - 图3

  • JDK1.5之前创建新执行线程有两种方法:
    • 继承Thread类的方式
    • 实现Runnable接口的方式

2)继承Thread类

学习提示:查看JDK帮助文档

  • 自定义线程类继承Thread类。
  • 子类中重写Thread类中的run()方法,编写线程执行体。
  • 创建Thread子类对象,即创建了线程对象。
  • 调用线程对象的start()方法:启动线程,调用run方法。

注意:线程不一定立即执行,CPU安排调度
多线程 - 图4

  1. package com.wang.thread;
  2. //创建线程方式一:基础Thread类,重写run()方法,调用start开启线程
  3. public class ThreadTest01 extends Thread {
  4. @Override
  5. public void run() {
  6. //run()方法线程体
  7. for (int i = 0; i < 20; i++) {
  8. System.out.println("我在看代码--" + i);
  9. }
  10. }
  11. public static void main(String[] args) {
  12. //main()线程,主线程
  13. //创建一个线程对象
  14. ThreadTest01 threadTest01 = new ThreadTest01();
  15. //调用start()方法开启线程
  16. threadTest01.start();
  17. for (int i = 0; i < 200; i++) {
  18. System.out.println("我再学习多线程--" + i);
  19. }
  20. }
  21. }
  • 注意点:
    • 如果自己手动调用run()方法,那么就只是普通方法,没有启动多线程模式。
    • run()方法由JVM调用,什么时候调用,执行的过程控制都由操作系统的CPU调度决定。
    • 想要启动多线程,必须调用start方法。
    • 一个线程对象只能调用一次start()方法启动,如果重复调用了,则将抛出以上的异常“IllegalThreadStateException”。

案例:实现多线程下载图片

  1. package com.wang.thread;
  2. import org.apache.commons.io.FileUtils;
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.net.URL;
  6. //练习Thread,实现多线程同步下载图片
  7. public class ThreadTest02 extends Thread {
  8. private String url; //网络图片地址
  9. private String name; //保存的文件名
  10. public ThreadTest02(String url, String name) {
  11. this.url = url;
  12. this.name = name;
  13. }
  14. //重写run()方法
  15. @Override
  16. public void run() {
  17. //run()方法的执行体 - 下载图片线程的执行体
  18. WebDownloader webDownloader = new WebDownloader();
  19. webDownloader.downloader(url, name);
  20. System.out.println("下载的文件名为:" + name);
  21. }
  22. //主线程
  23. public static void main(String[] args) {
  24. ThreadTest02 t1 = new ThreadTest02("https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/70535af2cf4d4887ae52faa218600606~tplv-k3u1fbpfcp-no-mark:480:400:0:0.awebp?", "1.jpg");
  25. ThreadTest02 t2 = new ThreadTest02("https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/08787c0586aa4ee680de770873deb0c6~tplv-k3u1fbpfcp-zoom-crop-mark:1304:1304:1304:734.awebp", "2.jpg");
  26. ThreadTest02 t3 = new ThreadTest02("https://pic1.zhimg.com/v2-19effed5b68c877608d720d205c6f37d_1440w.jpg?source=172ae18b", "3.jpg");
  27. //调用start()方法开启线程
  28. t1.start();
  29. t2.start();
  30. t3.start();
  31. }
  32. }
  33. //下载器
  34. class WebDownloader {
  35. //下载方法
  36. public void downloader(String url, String name) {
  37. try {
  38. FileUtils.copyURLToFile(new URL(url), new File(name));
  39. } catch (IOException e) {
  40. e.printStackTrace();
  41. System.out.println("IO异常,downloader方法出现问题!");
  42. }
  43. }
  44. }

多线程 - 图5
运行结果:
多线程 - 图6

Thread类的有关方法

  • start(): 启动线程,并执行对象的run()方法
  • run(): 线程在被调度时执行的操作
  • String getName(): 返回线程的名称
  • void setName(String name): 设置该线程名称
  • static Thread currentThread(): 返回当前线程。在Thread子类中就是this,通常用于主线程和Runnable实现类
  • static void yield(): 线程让步
    • 暂停当前正在执行的线程,把执行机会让给优先级相同或更高的线程
    • 若队列中没有同优先级的线程,忽略此方法
  • join():当某个程序执行流中调用其他线程的join() 方法时,调用线程将被阻塞,直到join() 方法加入的join线程执行完为止
    • 低优先级的线程也可以获得执行
  • static void sleep(long millis):(指定时间:毫秒)
    • 令当前活动线程在指定时间段内放弃对CPU控制,使其他线程有机会被执行,时间到后重排队。
    • 抛出InterruptedException异常
  • stop(): 强制线程生命期结束,不推荐使用。
  • booleanisAlive():返回boolean,判断线程是否还活着

3)实现Runable接口

学习提示:查看JDK帮助文档

  • 定义子类,实现Runnable接口。
  • 子类中重写Runnable接口中的run方法。
  • 通过Thread类含参构造器创建线程对象。
  • 将Runnable接口的子类对象作为实际参数传递给Thread类的构造器中。
  • 调用Thread类的start方法:开启线程,调用Runnable子类接口的run方法。

    推荐使用Runnable对象,因为Java单继承的局限性

  1. package com.wang.thread;
  2. //创建线程方式二: 实现runnable接口,重写run方法,执行线程需要丢入runnable接口实现类,调用start方法开启线程。
  3. public class ThreadTest03 implements Runnable {
  4. @Override
  5. public void run() {
  6. //run()方法线程体
  7. for (int i = 0; i < 20; i++) {
  8. System.out.println("我在看代码--" + i);
  9. }
  10. }
  11. public static void main(String[] args) {
  12. //main()线程,主线程
  13. //创建runnable接口的实现类对象
  14. ThreadTest03 threadTest03 = new ThreadTest03();
  15. //创建线程对象,通过线程对象来开启线程 -- 代理
  16. //Thread thread = new Thread(threadTest03);
  17. //thread.start();
  18. new Thread(threadTest03).start();
  19. for (int i = 0; i < 200; i++) {
  20. System.out.println("我再学习多线程--" + i);
  21. }
  22. }
  23. }

案例:用Runnable接口实现多线程下载图片

  1. package com.wang.thread;
  2. import org.apache.commons.io.FileUtils;
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.net.URL;
  6. //用Runnable接口实现多线程同步下载图片
  7. public class ThreadTest04 implements Runnable {
  8. private String url; //网络图片地址
  9. private String name; //保存的文件名
  10. public ThreadTest04(String url, String name) {
  11. this.url = url;
  12. this.name = name;
  13. }
  14. //重写run()方法
  15. @Override
  16. public void run() {
  17. //run()方法的执行体 - 下载图片线程的执行体
  18. WebDownloader01 webDownloader = new WebDownloader01();
  19. webDownloader.downloader(url, name);
  20. System.out.println("下载的文件名为:" + name);
  21. }
  22. //主线程
  23. public static void main(String[] args) {
  24. ThreadTest02 t1 = new ThreadTest02("https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/70535af2cf4d4887ae52faa218600606~tplv-k3u1fbpfcp-no-mark:480:400:0:0.awebp?", "1.jpg");
  25. ThreadTest02 t2 = new ThreadTest02("https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/08787c0586aa4ee680de770873deb0c6~tplv-k3u1fbpfcp-zoom-crop-mark:1304:1304:1304:734.awebp", "2.jpg");
  26. ThreadTest02 t3 = new ThreadTest02("https://pic1.zhimg.com/v2-19effed5b68c877608d720d205c6f37d_1440w.jpg?source=172ae18b", "3.jpg");
  27. //调用start()方法开启线程
  28. new Thread(t1).start();
  29. new Thread(t2).start();
  30. new Thread(t3).start();
  31. }
  32. }
  33. //下载器
  34. class WebDownloader01 {
  35. //下载方法
  36. public void downloader(String url, String name) {
  37. try {
  38. FileUtils.copyURLToFile(new URL(url), new File(name));
  39. } catch (IOException e) {
  40. e.printStackTrace();
  41. System.out.println("IO异常,downloader方法出现问题!");
  42. }
  43. }
  44. }

小结

  • 继承Thread类
    • 子类继承Thread类具备多线程能力
    • 启动线程:子类对象. start()
    • 不建议使用:避免OOP单继承局限性
  • 实现Runnable接口
    • 实现接口Runnable具有多线程能力
    • 启动线程:传入目标对象+Thread对象.start()
    • 推荐使用:避免单继承局限性,灵活方便,方便同一个对象被多个线程使用

继承方式和实现方式的联系与区别
**public class Thread extends Objectimplements Runnable**

  • 区别
    • 继承Thread:线程代码存放Thread子类run方法中。
    • 实现Runnable:线程代码存在接口的子类的run方法。
  • 实现方式的好处
    • 避免了单继承的局限性
    • 多个线程可以共享同一个接口实现类的对象,非常适合多个相同线程来处理同一份资源。
  1. package com.wang.thread;
  2. //多个线程操作同一个对象
  3. //买火车票的例子
  4. //发现问题:多个线程操作同一个资源的情况下,线程不安全,数据紊乱。
  5. public class ThreadTest05 implements Runnable {
  6. //票数
  7. private int ticketNums = 10;
  8. //重写run方法
  9. @Override
  10. public void run() {
  11. while (true) {
  12. if (ticketNums <= 0) {
  13. break;
  14. }
  15. //模拟延时
  16. try {
  17. Thread.sleep(200);
  18. } catch (InterruptedException e) {
  19. e.printStackTrace();
  20. }
  21. System.out.println(Thread.currentThread().getName() + "-->拿到了第" + ticketNums-- + "张票");
  22. }
  23. }
  24. public static void main(String[] args) {
  25. ThreadTest05 ticket = new ThreadTest05();
  26. new Thread(ticket, "小明").start();
  27. new Thread(ticket, "老师").start();
  28. new Thread(ticket, "黄牛党").start();
  29. }
  30. }

多线程 - 图7
发现问题:多个线程操作同一个资源的情况下,线程不安全,数据紊乱。 —— 解决:线程同步

案例:龟兔赛跑- Race

  • 首先来个赛道距离,然后要离终点越来越近
  • 判断比赛是否结束
  • 打印出胜利者
  • 龟兔赛跑开始
  • 故事中是乌龟赢的,兔子需要睡觉,所以需要模拟兔子睡觉
  • 最终,乌龟赢的比赛。 ```java package com.wang.thread;

//模拟龟兔赛跑 public class Race implements Runnable { //胜利者 private static String winner;

  1. @Override
  2. public void run() {
  3. for (int i = 0; i <= 100; i++) {
  4. if (Thread.currentThread().getName().equals("兔子") && i % 10 == 0) {
  5. try {
  6. Thread.sleep(10);
  7. } catch (InterruptedException e) {
  8. e.printStackTrace();
  9. }
  10. }
  11. //判断比赛是否结束
  12. boolean flag = gameOver(i);
  13. //如果比赛结束了,就停止程序
  14. if (flag) {
  15. break;
  16. }
  17. System.out.println(Thread.currentThread().getName() + "-->跑了" + i + "步");
  18. }
  19. }
  20. //判断是否完成比赛
  21. private boolean gameOver(int steps) {
  22. //判断是否有胜利者
  23. if (winner != null) {//已经存在胜利者了
  24. return true;
  25. }{
  26. if (steps >= 100) {
  27. winner = Thread.currentThread().getName();
  28. System.out.println("Winner is " + winner);
  29. return true;
  30. }
  31. }
  32. return false;
  33. }
  34. //主线程
  35. public static void main(String[] args) {
  36. Race race = new Race();
  37. //调用start方法开启线程
  38. new Thread(race, "乌龟").start();
  39. new Thread(race, "兔子").start();
  40. }

}

  1. <a name="059f3bff"></a>
  2. ### 4)实现Callable接口(了解即可)
  3. 1. 实现Callable接口,需要返回值类型
  4. 2. 重写call方法,需要抛出异常
  5. 3. 创建目标对象
  6. 4. 创建执行服务:`**ExecutorService ser = Executors.newFixedThreadPool(1);**`
  7. 5. 提交执行:`**Future<Boolean> result1 = ser.submit(t1);**`
  8. 6. 获取结果:`**boolean r1 = result1.get()**`
  9. 7. 关闭服务:`**ser.shutdownNow();**`
  10. ```java
  11. package com.wang.thread.callable;
  12. import org.apache.commons.io.FileUtils;
  13. import java.io.File;
  14. import java.io.IOException;
  15. import java.net.URL;
  16. import java.util.concurrent.*;
  17. //线程创建方式三:实现callable接口
  18. /*
  19. callable好处:
  20. 1.可以定义返回值
  21. 2.可以抛出异常
  22. */
  23. public class CallableTest implements Callable<Boolean> {
  24. private String url; //网络图片地址
  25. private String name; //保存的文件名
  26. public CallableTest(String url, String name) {
  27. this.url = url;
  28. this.name = name;
  29. }
  30. //下载图片线程的执行体
  31. @Override
  32. public Boolean call() throws Exception {
  33. WebDownloader02 webDownloader = new WebDownloader02();
  34. webDownloader.download(url, name);
  35. System.out.println("下载的文件名为:" + name);
  36. return true;
  37. }
  38. public static void main(String[] args) throws ExecutionException, InterruptedException {
  39. CallableTest t1 = new CallableTest("https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/70535af2cf4d4887ae52faa218600606~tplv-k3u1fbpfcp-no-mark:480:400:0:0.awebp?", "1.jpg");
  40. CallableTest t2 = new CallableTest("https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/08787c0586aa4ee680de770873deb0c6~tplv-k3u1fbpfcp-zoom-crop-mark:1304:1304:1304:734.awebp", "2.jpg");
  41. CallableTest t3 = new CallableTest("https://pic1.zhimg.com/v2-19effed5b68c877608d720d205c6f37d_1440w.jpg?source=172ae18b", "3.jpg");
  42. //创建执行服务
  43. ExecutorService ser = Executors.newFixedThreadPool(3);
  44. //提交执行
  45. Future<Boolean> r1 = ser.submit(t1);
  46. Future<Boolean> r2 = ser.submit(t2);
  47. Future<Boolean> r3 = ser.submit(t3);
  48. //获取结果
  49. Boolean rs1 = r1.get();
  50. Boolean rs2 = r2.get();
  51. Boolean rs3 = r3.get();
  52. //关闭服务
  53. ser.shutdown();
  54. }
  55. }
  56. //下载器
  57. class WebDownloader02 {
  58. //下载方法
  59. public void download(String url, String name) {
  60. try {
  61. FileUtils.copyURLToFile(new URL(url), new File(name));
  62. } catch (IOException e) {
  63. e.printStackTrace();
  64. System.out.println("IO异常,downloader方法出现问题!");
  65. }
  66. }
  67. }

5)静态代理模式

案例:实现静态代理对比Thread

  1. package com.wang.thread;
  2. public class StaticProxy {
  3. public static void main(String[] args) {
  4. You you = new You();
  5. new Thread(() -> System.out.println("我爱你")).start(); //lambda表达式
  6. new WeddingCompany(new You()).HappyMarry();
  7. }
  8. }
  9. interface Marry {
  10. void HappyMarry();
  11. }
  12. //真实角色
  13. class You implements Marry {
  14. @Override
  15. public void HappyMarry() {
  16. System.out.println("要结婚了,超开心!");
  17. }
  18. }
  19. //代理角色
  20. class WeddingCompany implements Marry {
  21. private Marry target;
  22. public WeddingCompany(Marry target) {
  23. this.target = target;
  24. }
  25. @Override
  26. public void HappyMarry() {
  27. before();
  28. this.target.HappyMarry();
  29. after();
  30. }
  31. private void after() {
  32. System.out.println("结婚之后,收尾款!");
  33. }
  34. private void before() {
  35. System.out.println("结婚之前,布置现场!");
  36. }
  37. }

多线程 - 图8

静态代理模式总结:

  • 真实对象和代理对象都要实现同一个接口
  • 代理对象要代理真实角色

好处:

  • 代理对象可以做很多真实对象做不了的事情
  • 真实对象专注于做自己的事情

6)Lamda表达式

  • λ 希腊字母表中排序第十一位的字母,英语名称为Lambda
  • 避免匿名内部类定义过多
  • 其实质属于函数式编程的概念
    1. params)-> expression [ 表达式 ]
    2. params)-> statement [ 语句 ]
    3. params)-> { statements }
    多线程 - 图9

为什么要使用lambda表达式?

  • 避免匿名内部类定义过多
  • 让代码看起来很简洁
  • 去掉没有意义的代码,只留下核心的逻辑
  • 理解Functional Interface(函数式接口)是学习Java8 lambda表达式的关键所在。
  • 函数式接口的定义:

    • 任何接口,如果只包含唯一一个抽象方法,那么它就是一个函数式接口。

      1. public interface Runnable {
      2. public abstract void run();
      3. }
    • 对于函数式接口,可以通过lambda表达式来创建该接口的对象。

代码推导lambda表达式

  1. package com.wang.lambda;
  2. //推导lambda表达式
  3. public class LambdaTest {
  4. //3.静态内部类
  5. static class Like2 implements ILike {
  6. @Override
  7. public void lambda() {
  8. System.out.println("I like lambda2");
  9. }
  10. }
  11. public static void main(String[] args) {
  12. ILike like = new Like();
  13. like.lambda();
  14. like = new Like2();
  15. like.lambda();
  16. //4.局部内部类
  17. class Like3 implements ILike {
  18. @Override
  19. public void lambda() {
  20. System.out.println("I like lambda3");
  21. }
  22. }
  23. like = new Like3();
  24. like.lambda();
  25. //5.匿名内部类 --类没有名称,必须借助接口或父类
  26. like = new ILike() {
  27. @Override
  28. public void lambda() {
  29. System.out.println("I like lambda4");
  30. }
  31. };
  32. like.lambda();
  33. //6.用lambda简化
  34. like = () -> {
  35. System.out.println("I like lambda5");
  36. };
  37. like.lambda();
  38. }
  39. }
  40. //1.定义一个函数式接口
  41. interface ILike {
  42. void lambda();
  43. }
  44. //2.实现类
  45. class Like implements ILike {
  46. @Override
  47. public void lambda() {
  48. System.out.println("I like lambda");
  49. }
  50. }

多线程 - 图10

案例

  1. package com.wang.lambda;
  2. public class LambdaTest02 {
  3. public static void main(String[] args) {
  4. ILove love = null;
  5. //1.lambda表达式简化
  6. love = (int a, int b, int c) -> {
  7. System.out.println("I love you -->" + a);
  8. System.out.println("I love you -->" + b);
  9. System.out.println("I love you -->" + c);
  10. };
  11. //简化1:去掉参数类型
  12. love = (a, b, c) -> {
  13. System.out.println("I love you -->" + a);
  14. System.out.println("I love you -->" + b);
  15. System.out.println("I love you -->" + c);
  16. };
  17. //简化2:简化括号
  18. //love = a -> {
  19. // System.out.println("I love you -->" + a);
  20. //};
  21. //简化3:去掉花括号
  22. //love = a -> System.out.println("I love you -->" + a);
  23. love.love(520, 520, 250);
  24. }
  25. }
  26. //定义一个函数式接口
  27. interface ILove {
  28. void love(int a, int b, int c);
  29. }

多线程 - 图11

总结:

  • lambda表达式只能有一行代码的情况下才能简化成为一行,如果有多行,那么就用代码块{}包裹;
  • 前提是接口为函数式接口。
  • 多个参数也可以去掉参数类型,要去掉就都去掉,必须加上括号。

线程的生命周期

1)线程状态

  • JDK中用Thread.State类定义了线程的几种状态
    要想实现多线程,必须在主线程中创建新的线程对象。Java语言使用Thread类及其子类的对象来表示线程,在它的一个完整的生命周期中通常要经历如下的五种状态:
    • 新建:当一个Thread类或其子类的对象被声明并创建时,新生的线程对象处于新建状态
    • 就绪:处于新建状态的线程被start()后,将进入线程队列等待CPU时间片,此时它已具备了运行的条件,只是没分配到CPU资源
    • 运行:当就绪的线程被调度并获得CPU资源时,便进入运行状态,run()方法定义了线程的操作和功能
    • 阻塞:在某种特殊情况下,被人为挂起或执行输入输出操作时,让出CPU并临时中止自己的执行,进入阻塞状态
    • 死亡:线程完成了它的全部工作或线程被提前强制性地中止或出现异常导致结束

多线程 - 图12
多线程 - 图13
多线程 - 图14

2)线程方法

  • setPriority(int newPriority) :更改线程的优先级
  • static void sleep(long millis) :在指定的毫秒数内让当前正在执行的线程休眠
  • void join() :等待该线程终止
  • static void yield() :暂停当前正在执行的线程对象,并执行其他线程
  • void interrupt() :中断线程,别用这个方式
  • boolean isAlive() :测试线程是否处于活动状态

3)停止线程

  • 不推荐使用JDK提供的 stop()、destroy()方法。【已废弃】
  • 推荐线程自己停止下来
  • 建议使用一个标志位进行终止变量
    • 当flag=false,则终止线程运行。 ```java package com.wang.thread.state;

//测试停止线程 //1.建议线程正常停止 ——> 利用次数,不建议死循环 //2.建议使用标志位 ——> 设置一个标志位 //3.不要使用stop 或者destroy等过时或JDK不建议使用的方法

public class StopTest implements Runnable { //1.设置一个标识位 private boolean flag = true;

  1. @Override
  2. public void run() {
  3. int i = 0;
  4. while (true) {
  5. System.out.println("run...Thread" + i);
  6. }
  7. }
  8. //2.设置一个公开的方法停止线程,转换标识位
  9. public void stop() {
  10. this.flag = false;
  11. }
  12. public static void main(String[] args) {
  13. StopTest stopTest = new StopTest();
  14. new Thread(stopTest).start();
  15. for (int i = 0; i < 1000; i++) {
  16. System.out.println("main --> " + i);
  17. if (i == 900) {
  18. //调用stop方法切换标识位,让线程停止
  19. stopTest.stop();
  20. System.out.println("线程该停止了!");
  21. }
  22. }
  23. }

}

  1. <a name="b2390e61"></a>
  2. ### 4)线程休眠-sleep
  3. - sleep (时间) 指定当前线程阻塞的毫秒数。(1000毫秒=1秒)
  4. - sleep存在异常InterruptedException。
  5. - sleep时间达到后线程进入就绪状态。
  6. - sleep可以模拟网络延时、倒计时等。
  7. - **每一个对象都有一个锁,sleep不会释放锁。**
  8. **演示:模拟网络延时**
  9. ```java
  10. package com.wang.thread.sleep;
  11. //模拟网络延时:放大问题的发生性
  12. public class SleepTest implements Runnable{
  13. //票数
  14. private int tickNums =10;
  15. @Override
  16. public void run() {
  17. while (true){
  18. if (tickNums<=0){
  19. break;
  20. }
  21. //模拟网络延时
  22. try {
  23. Thread.sleep(100);
  24. } catch (InterruptedException e) {
  25. e.printStackTrace();
  26. }
  27. System.out.println(Thread.currentThread().getName()+" --> 拿到了第"+tickNums--+"张票");
  28. }
  29. }
  30. public static void main(String[] args) {
  31. SleepTest ticket = new SleepTest();
  32. new Thread(ticket,"小明").start();
  33. new Thread(ticket," 老师").start();
  34. new Thread(ticket,"黄牛").start();
  35. }
  36. }

演示:模拟倒计时

  1. package com.wang.thread.sleep;
  2. //模拟倒计时
  3. public class SleepTest02 {
  4. public static void main(String[] args) {
  5. try {
  6. tenDown();
  7. } catch (InterruptedException e) {
  8. e.printStackTrace();
  9. }
  10. }
  11. //10秒倒计时
  12. public static void tenDown() throws InterruptedException {
  13. int num = 10;
  14. while (true) {
  15. Thread.sleep(1000);
  16. System.out.println(num--);
  17. if (num <= 0) {
  18. break;
  19. }
  20. }
  21. }
  22. }

多线程 - 图15

演示:打印系统当前时间

  1. package com.wang.thread.sleep;
  2. import java.text.SimpleDateFormat;
  3. import java.util.Date;
  4. public class SleepTest03 {
  5. public static void main(String[] args) {
  6. //打印当前系统时间
  7. Date startTime = new Date(System.currentTimeMillis());//获取系统当前时间
  8. while (true){
  9. try {
  10. Thread.sleep(1000);
  11. System.out.println(new SimpleDateFormat("HH:mm:ss").format(startTime));
  12. startTime = new Date(System.currentTimeMillis());//更新当前时间
  13. } catch (InterruptedException e) {
  14. e.printStackTrace();
  15. }
  16. }
  17. }
  18. }

多线程 - 图16

5)线程礼让-yield

  • 礼让线程,让当前正在执行的线程暂停,但不阻塞
  • 将线程从运行状态转为就绪状态
  • 让cpu重新调度,礼让不一定成功!看CPU心情 ```java package com.wang.thread.yield;

//测试礼让线程 //由CPU重新调度,礼让不一定成功,看CPU心情 public class YieldTest { public static void main(String[] args) { MyYield myYield = new MyYield();

  1. new Thread(myYield, "a").start();
  2. new Thread(myYield, "b").start();
  3. }

}

class MyYield implements Runnable {

  1. @Override
  2. public void run() {
  3. System.out.println(Thread.currentThread().getName() + "线程开始执行");
  4. Thread.yield();//线程礼让
  5. System.out.println(Thread.currentThread().getName() + "线程停止执行");
  6. }

}

  1. ![](https://gitee.com/wang_jin0751/image-host/raw/master/202205051414693.png#crop=0&crop=0&crop=1&crop=1&height=199&id=U72F9&originHeight=398&originWidth=1660&originalType=binary&ratio=1&rotation=0&showTitle=false&status=done&style=none&title=&width=830)<br />礼让成功。<br />![](https://gitee.com/wang_jin0751/image-host/raw/master/202205051416887.png#crop=0&crop=0&crop=1&crop=1&height=143&id=M2CKj&originHeight=286&originWidth=1506&originalType=binary&ratio=1&rotation=0&showTitle=false&status=done&style=none&title=&width=753)
  2. 没有礼让成功。
  3. <a name="bccc1b09"></a>
  4. ### 6)join
  5. - Join合并线程,待此线程执行完成后,再执行其他线程,其他线程阻塞
  6. - 可以想象成插队
  7. ```java
  8. package com.wang.thread.join;
  9. //测试join方法
  10. //想象为插队
  11. public class JoinTest implements Runnable {
  12. @Override
  13. public void run() {
  14. for (int i = 0; i < 100; i++) {
  15. System.out.println("线程VIP来了" + i);
  16. }
  17. }
  18. public static void main(String[] args) throws InterruptedException {
  19. //启动我们的线程
  20. JoinTest joinTest = new JoinTest();
  21. Thread thread = new Thread(joinTest);
  22. thread.start();
  23. //主线程
  24. for (int i = 0; i < 500; i++) {
  25. if (i == 200) {
  26. thread.join();//插队
  27. }
  28. System.out.println("main" + i);
  29. }
  30. }
  31. }

7)线程状态观测

  • Thread.State

查看JDK帮助文档
多线程 - 图17

  1. package com.wang.thread.state;
  2. //线程状态的观测
  3. public class StateTest {
  4. public static void main(String[] args) {
  5. Thread thread = new Thread(() -> {
  6. for (int i = 0; i < 5; i++) {
  7. try {
  8. Thread.sleep(200);
  9. } catch (InterruptedException e) {
  10. e.printStackTrace();
  11. }
  12. }
  13. System.out.println("////");
  14. });
  15. //观察状态
  16. Thread.State state = thread.getState();
  17. System.out.println(state); //NEW
  18. //观察启动后
  19. thread.start();//启动线程
  20. state = thread.getState();
  21. System.out.println(state);//RUN
  22. while (state != Thread.State.TERMINATED) {//只要线程不终止,就一直输出线程状态
  23. try {
  24. Thread.sleep(100);
  25. } catch (InterruptedException e) {
  26. e.printStackTrace();
  27. }
  28. state = thread.getState();//更新线程状态
  29. System.out.println(state);//输出线程状态
  30. }
  31. }
  32. }

多线程 - 图18

8)线程的调度

  • 调度策略
    • 时间片

多线程 - 图19

  • 抢占式:高优先级的线程抢占CPU
    • Java的调度方法
  • 同优先级线程组成先进先出队列(先到先服务),使用时间片策略
  • 对高优先级,使用优先调度的抢占式策略

9)线程的优先级

  • Java提供一个线程调度器来监控程序中启动后进入就绪状态的所有线程,线程调度器按照优先级决定应该调度哪个线程来执行。
  • 线程的优先级用数字表示,范围从1~10
    • MIN _PRIORITY1
    • MAX_PRIORITY10
    • NORM_PRIORITY5
  • 涉及的方法
    • getPriority() :返回线程优先值
    • setPriority(intnewPriority) :改变线程的优先级
  • 说明
    • 优先级的设定建议在start()调度前
    • 线程创建时继承父线程的优先级
    • 低优先级只是意味着获得调度的概率低,并非优先级低的一定是在高优先级线程之后才被调用,这取决于CPU的调度 ```java package com.wang.thread;

//线程优先级 public class PriorityTest { public static void main(String[] args) {

  1. //主线程默认优先级
  2. System.out.println(Thread.currentThread().getName() + "-->" + Thread.currentThread().getPriority());
  3. MyPriority myPriority = new MyPriority();
  4. Thread t1 = new Thread(myPriority);
  5. Thread t2 = new Thread(myPriority);
  6. Thread t3 = new Thread(myPriority);
  7. Thread t4 = new Thread(myPriority);
  8. Thread t5 = new Thread(myPriority);
  9. Thread t6 = new Thread(myPriority);
  10. //先设置优先级,再启动
  11. t1.start();
  12. t2.setPriority(1);
  13. t2.start();
  14. t3.setPriority(4);
  15. t3.start();
  16. t4.setPriority(Thread.MIN_PRIORITY);//MIN_PRIORITY=1
  17. t4.start();
  18. t5.setPriority(Thread.MAX_PRIORITY);//MAX_PRIORITY=10
  19. t5.start();
  20. t6.setPriority(8);
  21. t6.start();
  22. }

}

class MyPriority implements Runnable {

  1. @Override
  2. public void run() {
  3. System.out.println(Thread.currentThread().getName() + "-->" + Thread.currentThread().getPriority());
  4. }

}

  1. <a name="852abfe1"></a>
  2. ### 10)线程的分类
  3. Java中的线程分为两类:一种是**守护(daemon)线程**,一种是**用户线程**。它们在几乎每个方面都是相同的,唯一的区别是判断JVM何时离开。
  4. - 虚拟机必须确保用户线程执行完毕,不用等待守护线程执行完毕。
  5. - 守护线程是用来服务用户线程的,通过在start()方法前调用**thread.setDaemon(true**)可以把一个用户线程变成一个守护线程。
  6. - Java垃圾回收就是一个典型的守护线程。
  7. - 若JVM中都是守护线程,当前JVM将退出。
  8. ```java
  9. package com.wang.thread;
  10. //守护线程
  11. //上帝守护我们
  12. public class DaemonTest {
  13. public static void main(String[] args) {
  14. God god = new God();
  15. We we = new We();
  16. Thread thread = new Thread(god);
  17. thread.setDaemon(true);//默认是false表示是用户线程,正常的线程都是用户线程
  18. thread.start();//上帝守护线程启动
  19. new Thread(we).start();//用户线程启动
  20. }
  21. }
  22. //上帝
  23. class God implements Runnable {
  24. @Override
  25. public void run() {
  26. while (true) {
  27. System.out.println("上帝保佑着我们");
  28. }
  29. }
  30. }
  31. //你
  32. class We implements Runnable {
  33. @Override
  34. public void run() {
  35. for (int i = 0; i < 36500; i++) {
  36. System.out.println("我们一生开心地活着");
  37. }
  38. System.out.println("====goodbye world!====");
  39. }
  40. }

线程同步(重点)

  • 线程同步:多个线程操作同一个资源。
  • 并发 : 同一个对象被多个线程同时操作
  • 处理多线程问题时 , 多个线程访问同一个对象(兵法) , 并且某些线程还想修改这个对象,这时候就需要线程同步。线程同步其实就是一种等待机制 , 多个需要同时访问此对象的线程进入这个对象的等待池形成队列, 等待前面线程使用完毕 , 下一个线程再使用。
  • 队列和锁保证线程同步的安全性
  • 问题的提出
    • 多个线程执行的不确定性引起执行结果的不稳定
    • 多个线程对账本的共享,会造成操作的不完整性,会破坏数据。

例题:模拟火车站售票程序,开启三个窗口售票。
多线程 - 图20
多线程 - 图21
多线程 - 图22

  1. private int tick = 100;
  2. public void run(){
  3. while(true){
  4. if(tick>0){
  5. try{
  6. Thread.sleep(10);
  7. }catch(InterruptedException e){ e.printStackTrace();}
  8. System.out.println(Thread.currentThread().getName()+"售出车票,tick号为:"+ tick--);
  9. }
  10. }
  11. }
  • 多线程出现了安全问题
  • 问题的原因:
    • 当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,另一个线程参与进来执行,导致共享数据的错误。
  • 解决办法:
    • 对多条操作共享数据的语句,只能让一个线程都执行完,在执行过程中,其他线程不可以参与执行。

1)线程同步

  • 由于同一进程的多个线程共享同一块存储空间 , 在带来方便的同时,也带来访问冲突问题 , 为了保证数据在方法中被访问时的正确性 , 在访问时加入 锁机制synchronized 。当一个线程获得对象的排它锁 , 独占资源 , 其他线程必须等待,使用后释放锁即可。存在以下问题 :
    • 一个线程持有锁会导致其他所有需要此锁的线程挂起。
    • 在多线程竞争下 , 加锁 , 释放锁会导致比较多的上下文切换 和 调度延时,引起性能问题。
    • 如果一个优先级高的线程等待一个优先级低的线程释放锁 会导致优先级倒置 , 引起性能问题。
  • Java对于多线程的安全问题提供了专业的解决方式:同步机制

三大不安全案例
1、不安全的买票问题

  1. package com.wang.thread.syn;
  2. //不安全的买票
  3. public class UnsafeBuyTicket {
  4. public static void main(String[] args) {
  5. BuyTicket station = new BuyTicket();
  6. new Thread(station, "苦逼的我").start();
  7. new Thread(station, "牛逼的你们").start();
  8. new Thread(station, "可恶的黄牛党").start();
  9. }
  10. }
  11. class BuyTicket implements Runnable {
  12. private int ticketNums = 10;//票数
  13. boolean flag = true;//外部停止标识位
  14. @Override
  15. public void run() {
  16. //买票
  17. while (flag) {
  18. try {
  19. buy();
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. }
  23. }
  24. }
  25. private void buy() throws InterruptedException {
  26. //判断是否有票
  27. if (ticketNums <= 0) {
  28. flag = false;
  29. return;
  30. }
  31. //模拟延时
  32. Thread.sleep(100);
  33. //买票
  34. System.out.println(Thread.currentThread().getName() + "拿到了第" + ticketNums-- + "张票");
  35. }
  36. }

多线程 - 图23
线程不安全:

  • 出现同一张票被不同的人拿到的
  • 出现负数

2、不安全的银行取钱问题

  1. package com.wang.thread.syn;
  2. //两个人取钱
  3. //需要账户:余额、卡名
  4. public class UnsafeBank {
  5. public static void main(String[] args) {
  6. //账户
  7. Account account = new Account(100, "结婚基金");
  8. Drawing you = new Drawing(account, 50, "you");
  9. Drawing girlFriend = new Drawing(account, 100, "girlFriend");
  10. you.start();
  11. girlFriend.start();
  12. }
  13. }
  14. //账户
  15. class Account {
  16. int money;
  17. String name;
  18. public Account(int money, String name) {
  19. this.money = money;
  20. this.name = name;
  21. }
  22. }
  23. //银行:模拟取款
  24. class Drawing extends Thread {
  25. //账户
  26. Account account;
  27. //取了多少钱
  28. int drawingMoney;
  29. //现在手里有多少钱
  30. int nowMoney;
  31. public Drawing(Account account, int drawingMoney, String name) {
  32. super(name);
  33. this.account = account;
  34. this.drawingMoney = drawingMoney;
  35. }
  36. //取钱
  37. @Override
  38. public void run() {
  39. //判断有没有钱
  40. if (account.money - drawingMoney < 0) {
  41. System.out.println(Thread.currentThread().getName() + "钱不够了,取不了了");
  42. return;
  43. }
  44. //模拟延时
  45. //sleep可以放大问题的发生性
  46. try {
  47. Thread.sleep(1000);
  48. } catch (InterruptedException e) {
  49. e.printStackTrace();
  50. }
  51. //卡内的余额 = 余额 - 取的钱
  52. account.money = account.money - drawingMoney;
  53. //你手里的钱
  54. nowMoney = nowMoney + drawingMoney;
  55. System.out.println(account.name + "余额为:" + account.money);
  56. //Thread.currentThread().getName() = this.getName()
  57. System.out.println(this.getName() + "手里的钱:" + nowMoney);
  58. }
  59. }

多线程 - 图24

3、线程不安全的集合

  1. package com.wang.thread.syn;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. //线程不安全的集合
  5. public class UnsafeList {
  6. public static void main(String[] args) {
  7. List<String> list = new ArrayList<String>();
  8. for (int i = 0; i < 10000; i++) {
  9. new Thread(() -> {
  10. list.add(Thread.currentThread().getName());
  11. }).start();
  12. }
  13. try {
  14. Thread.sleep(3000);
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. System.out.println(list.size());
  19. }
  20. }

2)同步方法

  • 由于可以通过 private 关键字来保证数据对象只能被方法访问 , 所以只需要针对方法提出一套机制 , 这套机制就是synchronized关键字 , 它包括两种用法 :synchronized方法synchronized块
    • 同步方法 : **public synchronized void method(int args) {}**
  • synchronized方法控制对 “对象” 的访问 , 每个对象对应一把锁 , 每个synchronized方法都必须获得调用该方法的对象的锁才能执行 , 否则线程会阻塞 ,方法一旦执行 , 就独占该锁 , 直到该方法返回才释放锁 , 后面被阻塞的线程才能获得这个锁 , 继续执行。
    • 缺陷 : 若将一个大的方法申明为synchronized 将会影响效率

使用线程同步实现买票(synchronized关键字)

  1. package com.wang.thread.syn;
  2. public class BuyTicketDemo {
  3. public static void main(String[] args) {
  4. BuyTickets station = new BuyTickets();
  5. new Thread(station, "苦逼的我").start();
  6. new Thread(station, "牛逼的你们").start();
  7. new Thread(station, "可恶的黄牛党").start();
  8. }
  9. }
  10. class BuyTickets implements Runnable {
  11. private int ticketNums = 10;//票数
  12. boolean flag = true;//外部停止标识位
  13. @Override
  14. public void run() {
  15. //买票
  16. while (flag) {
  17. try {
  18. buy();
  19. } catch (InterruptedException e) {
  20. e.printStackTrace();
  21. }
  22. }
  23. }
  24. //synchronized 同步方法,锁的是this
  25. private synchronized void buy() throws InterruptedException {
  26. //判断是否有票
  27. if (ticketNums <= 0) {
  28. flag = false;
  29. return;
  30. }
  31. //模拟延时
  32. Thread.sleep(100);
  33. //买票
  34. System.out.println(Thread.currentThread().getName() + "拿到了第" + ticketNums-- + "张票");
  35. }
  36. }

多线程 - 图25

3)同步块

  • 同步块

    1. synchronized (obj) {
    2. //需要被同步的代码
    3. }
  • Obj 称之为 同步监视器

    • Obj 可以是任何对象 , 但推荐使用共享资源作为同步监视器
    • 同步方法中无需指定同步监视器 , 因为同步方法的同步监视器就是this , 就是这个对象本身 , 或者是 class [ 反射中讲解 ]
  • 同步监视器的执行过程
    • 第一个线程访问 , 锁定同步监视器 , 执行其中代码 .
    • 第二个线程访问 , 发现同步监视器被锁定 , 无法访问 .
    • 第一个线程访问完毕 , 解锁同步监视器 .
    • 第二个线程访问, 发现同步监视器没有锁 , 然后锁定并访问

同步块实现银行取钱

  1. package com.wang.thread.syn;
  2. //两个人取钱
  3. //需要账户:余额、卡名
  4. public class UnsafeBank {
  5. public static void main(String[] args) {
  6. //账户
  7. Account account = new Account(100, "结婚基金");
  8. Drawing you = new Drawing(account, 50, "you");
  9. Drawing girlFriend = new Drawing(account, 100, "girlFriend");
  10. you.start();
  11. girlFriend.start();
  12. }
  13. }
  14. //账户
  15. class Account {
  16. int money;
  17. String name;
  18. public Account(int money, String name) {
  19. this.money = money;
  20. this.name = name;
  21. }
  22. }
  23. //银行:模拟取款
  24. class Drawing extends Thread {
  25. //账户
  26. Account account;
  27. //取了多少钱
  28. int drawingMoney;
  29. //现在手里有多少钱
  30. int nowMoney;
  31. public Drawing(Account account, int drawingMoney, String name) {
  32. super(name);
  33. this.account = account;
  34. this.drawingMoney = drawingMoney;
  35. }
  36. //取钱
  37. @Override
  38. public void run() {
  39. synchronized (account) {
  40. //判断有没有钱
  41. if (account.money - drawingMoney < 0) {
  42. System.out.println(Thread.currentThread().getName() + "钱不够了,取不了了");
  43. return;
  44. }
  45. //模拟延时
  46. //sleep可以放大问题的发生性
  47. try {
  48. Thread.sleep(1000);
  49. } catch (InterruptedException e) {
  50. e.printStackTrace();
  51. }
  52. //卡内的余额 = 余额 - 取的钱
  53. account.money = account.money - drawingMoney;
  54. //你手里的钱
  55. nowMoney = nowMoney + drawingMoney;
  56. System.out.println(account.name + "余额为:" + account.money);
  57. //Thread.currentThread().getName() = this.getName()
  58. System.out.println(this.getName() + "手里的钱:" + nowMoney);
  59. }
  60. }
  61. }

多线程 - 图26

同步块实现集合

  1. package com.wang.thread.syn;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. public class UnsafeList {
  5. public static void main(String[] args) {
  6. List<String> list = new ArrayList<String>();
  7. for (int i = 0; i < 10000; i++) {
  8. new Thread(() -> {
  9. //synchronized同步块
  10. synchronized (list) {
  11. list.add(Thread.currentThread().getName());
  12. }
  13. }).start();
  14. }
  15. try {
  16. Thread.sleep(3000);
  17. } catch (InterruptedException e) {
  18. e.printStackTrace();
  19. }
  20. System.out.println(list.size());
  21. }
  22. }

多线程 - 图27

总结:

  • 加上Synchronized关键字的方法锁的是方法锁在类的本身
  • synchronized同步块可以锁任何对象,锁的对象就是变化的量,即需要增删改的量

补充:JUC安全类型的集合

  1. package com.wang.thread;
  2. import java.util.concurrent.CopyOnWriteArrayList;
  3. //JUC安全类型的集合
  4. public class JucTest {
  5. public static void main(String[] args) {
  6. CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<String>();
  7. for (int i = 0; i < 10000; i++) {
  8. new Thread(() -> {
  9. list.add(Thread.currentThread().getName());
  10. }).start();
  11. }
  12. try {
  13. Thread.sleep(3000);
  14. } catch (InterruptedException e) {
  15. e.printStackTrace();
  16. }
  17. System.out.println(list.size());
  18. }
  19. }

拓展阅读:
为什么线程安全的List推荐用CopyOnWriteArrayList,而不是Vector

4)同步机制中的锁Lock

  • 同步锁机制:
    在《Thinking in Java》中,是这么说的:对于并发工作,你需要某种方式来防止两个任务访问相同的资源(其实就是共享资源竞争)。防止这种冲突的方法就是当资源被一个任务使用时,在其上加锁。第一个访问某项资源的任务必须锁定这项资源,使其他任务在其被解锁之前,就无法访问它了,而在其被解锁之时,另一个任务就可以锁定并使用它了。
  • synchronized的锁是什么?
    • 任意对象都可以作为同步锁。所有对象都自动含有单一的锁(监视器)。
    • 同步方法的锁:静态方法(类名.class)、非静态方法(this)
    • 同步代码块:自己指定,很多时候也是指定为this或类名.class
  • 注意:
    • 必须确保使用同一个资源的多个线程共用一把锁,这个非常重要,否则就无法保证共享资源的安全
    • 一个线程类中的所有静态方法共用同一把锁(类名.class),所有非静态方法共用同一把锁(this),同步代码块(指定需谨慎)

同步的范围
1、如何找问题,即代码是否存在线程安全?(非常重要)
(1)明确哪些代码是多线程运行的代码
(2)明确多个线程是否有共享数据
(3)明确多线程运行代码中是否有多条语句操作共享数据

2、如何解决呢?(非常重要)
对多条操作共享数据的语句,只能让一个线程都执行完,在执行过程中,其他线程不可以参与执行。
即所有操作共享数据的这些语句都要放在同步范围中

3、切记:

  • 范围太小:没锁住所有有安全问题的代码
  • 范围太大:没发挥多线程的功能。

释放锁的操作

  • 当前线程的同步方法、同步代码块执行结束。
  • 当前线程在同步代码块、同步方法中遇到break、return终止了该代码块、该方法的继续执行。
  • 当前线程在同步代码块、同步方法中出现了未处理的Error或Exception,导致异常结束。
  • 当前线程在同步代码块、同步方法中执行了线程对象的wait()方法,当前线程暂停,并释放锁。

不会释放锁的操作

  • 线程执行同步代码块或同步方法时,程序调用Thread.sleep()、Thread.yield()方法暂停当前线程的执行
  • 线程执行同步代码块时,其他线程调用了该线程的suspend()方法将该线程挂起,该线程不会释放锁(同步监视器)。
    • 应尽量避免使用suspend()和resume()来控制线程

单例设计模式之懒汉式(线程安全)

  1. class Singleton {
  2. private static Singleton instance = null;
  3. private Singleton(){}
  4. public static Singleton getInstance(){
  5. if(instance==null){
  6. synchronized(
  7. Singleton.class
  8. if(instance == null){
  9. instance=new Singleton();
  10. }
  11. }
  12. }
  13. return instance;
  14. }
  15. }
  16. public class SingletonTest{
  17. public static void main(String[]{
  18. Singleton s1=Singleton.getInstance();
  19. Singleton s2=Singleton.getInstance();
  20. System.out.println(s1==s2);
  21. }
  22. }

5)死锁

  • 多个线程各自占有一些共享资源,并且互相等待其他线程占有的资源才能运行,而导致两个或多个线程都在等待对方释放资源,都停止执行的情形。某一同步块同时拥有“两个以上对象的锁”时,就可能发生“死锁”问题。
    • 出现死锁后,不会出现异常,不会出现提示,只是所有的线程都处于阻塞状态,无法继续
  • 解决方法
    • 专门的算法、原则
    • 尽量减少同步资源的定义
    • 尽量避免嵌套同步

死锁避免方法

  • 产生死锁的四个必要条件:
    • 互斥条件:一个资源每次只能被一个进程使用。
    • 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
    • 不剥夺条件 : 进程已获得的资源,在末使用完之前,不能强行剥夺。
    • 循环等待条件 : 若干进程之间形成一种头尾相接的循环等待资源关系。
  • 上面死锁的四个必要条件,只要想办法破其中的任意一个或多个条件就可以避免死锁发生 ```java package com.wang.thread;

//死锁 public class DeadLock { public static void main(String[] args) { Makeup g1 = new Makeup(0, “灰姑娘”); Makeup g2 = new Makeup(1, “白雪公主”);

  1. g1.start();
  2. g2.start();
  3. }

}

//口红 class Lipstick {}

//镜子 class Mirror {}

class Makeup extends Thread {

  1. //需要的资源只有一份,使用static来保证只有一份
  2. static Lipstick lipstick = new Lipstick();
  3. static Mirror mirror = new Mirror();
  4. int choice;//选择
  5. String girlName;//使用化妆品的人
  6. Makeup(int choice, String girlName) {
  7. this.choice = choice;
  8. this.girlName = girlName;
  9. }
  10. @Override
  11. public void run() {
  12. //化妆
  13. try {
  14. makeup();
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. }
  19. //化妆:互相持有对方的锁,就是需要拿到对方的资源
  20. public void makeup() throws InterruptedException {
  21. if (choice == 0) {
  22. synchronized (lipstick) {
  23. System.out.println(this.girlName + "获得口红的锁");
  24. Thread.sleep(1000);
  25. synchronized (mirror) {//一秒钟后想获得镜子
  26. System.out.println(this.girlName + "获得镜子的锁");
  27. }
  28. }
  29. } else {
  30. synchronized (mirror) {
  31. System.out.println(this.girlName + "获得镜子的锁");
  32. Thread.sleep(2000);
  33. synchronized (lipstick) {//两秒钟后想获得镜子
  34. System.out.println(this.girlName + "获得口红的锁");
  35. }
  36. }
  37. }
  38. }

}

  1. 程序未结束,双方僵持,即出现死锁现象。<br />![](https://gitee.com/wang_jin0751/image-host/raw/master/202205052016317.png#crop=0&crop=0&crop=1&crop=1&height=118&id=CbdAt&originHeight=236&originWidth=1696&originalType=binary&ratio=1&rotation=0&showTitle=false&status=done&style=none&title=&width=848)
  2. 解决方法:把锁拿出来,不要在某一同步块同时拥有“**两个以上对象的锁**”
  3. ```java
  4. package com.wang.thread;
  5. public class DeadLock {
  6. public static void main(String[] args) {
  7. Makeup g1 = new Makeup(0, "灰姑娘");
  8. Makeup g2 = new Makeup(1, "白雪公主");
  9. g1.start();
  10. g2.start();
  11. }
  12. }
  13. //口红
  14. class Lipstick {}
  15. //镜子
  16. class Mirror {}
  17. class Makeup extends Thread {
  18. //需要的资源只有一份,使用static来保证只有一份
  19. static Lipstick lipstick = new Lipstick();
  20. static Mirror mirror = new Mirror();
  21. int choice;//选择
  22. String girlName;//使用化妆品的人
  23. Makeup(int choice, String girlName) {
  24. this.choice = choice;
  25. this.girlName = girlName;
  26. }
  27. @Override
  28. public void run() {
  29. //化妆
  30. try {
  31. makeup();
  32. } catch (InterruptedException e) {
  33. e.printStackTrace();
  34. }
  35. }
  36. //化妆:互相持有对方的锁,就是需要拿到对方的资源
  37. public void makeup() throws InterruptedException {
  38. if (choice == 0) {
  39. synchronized (lipstick) {
  40. System.out.println(this.girlName + "获得口红的锁");
  41. Thread.sleep(1000);
  42. }
  43. //把锁拿出来,不要在某一同步块同时拥有“两个以上对象的锁”
  44. synchronized (mirror) {//一秒钟后想获得镜子
  45. System.out.println(this.girlName + "获得镜子的锁");
  46. }
  47. } else {
  48. synchronized (mirror) {
  49. System.out.println(this.girlName + "获得镜子的锁");
  50. Thread.sleep(2000);
  51. }
  52. //把锁拿出来,不要在某一同步块同时拥有“两个以上对象的锁”
  53. synchronized (lipstick) {//两秒钟后想获得镜子
  54. System.out.println(this.girlName + "获得口红的锁");
  55. }
  56. }
  57. }
  58. }

多线程 - 图28

6)Lock(锁)

  • 从JDK 5.0开始,Java提供了更强大的线程同步机制——通过显式定义同步锁对象来实现同步,同步锁使用Lock对象充当
  • java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的工具。锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前应先获得Lock对象。
  • ReentrantLock(可重入锁) 类实现了Lock,它拥有与 synchronized 相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock,可以显式加锁、释放锁。
    1. class A{
    2. private final ReentrantLock lock = new ReenTrantLock();
    3. public void m(){
    4. //加锁
    5. lock.lock();
    6. try{
    7. //保证线程安全的代码;
    8. }
    9. finally{
    10. //解锁
    11. lock.unlock();
    12. //如果同步代码有异常,要将unlock()写入finally语句块
    13. }
    14. }
    15. }

演示:

  1. package com.wang.thread;
  2. import java.util.concurrent.locks.ReentrantLock;
  3. //Lock锁
  4. public class LockTest {
  5. public static void main(String[] args) {
  6. Lock2Test lock2Test = new Lock2Test();
  7. new Thread(lock2Test).start();
  8. new Thread(lock2Test).start();
  9. new Thread(lock2Test).start();
  10. }
  11. }
  12. class Lock2Test implements Runnable {
  13. int ticketNums = 10;
  14. //定义Lock锁
  15. private final ReentrantLock lock = new ReentrantLock();
  16. @Override
  17. public void run() {
  18. while (true) {
  19. try {
  20. //加锁
  21. lock.lock();
  22. if (ticketNums > 0) {
  23. try {
  24. Thread.sleep(1000);
  25. } catch (InterruptedException e) {
  26. e.printStackTrace();
  27. }
  28. System.out.println(ticketNums--);
  29. } else {
  30. break;
  31. }
  32. } finally {
  33. //解锁
  34. lock.unlock();
  35. }
  36. }
  37. }
  38. }

多线程 - 图29

6)synchronized 与 Lock 的对比

  • Lock是显式锁(手动开启和关闭锁,别忘记关闭锁),synchronized是隐式锁,出了作用域自动释放。
  • Lock只有代码块锁,synchronized有代码块锁和方法锁。
  • 使用Lock锁,JVM将花费较少的时间来调度线程,性能更好,并且具有更好的扩展性(提供更多的子类)。
  • 优先使用顺序:
    • Lock > 同步代码块(已经进入了方法体,分配了相应资源)> 同步方法(在方法体之外)

线程协作

  • 应用场景 : 生产者和消费者问题
    • 假设仓库中只能存放一件产品 , 生产者将生产出来的产品放入仓库 , 消费者将仓库中产品取走消费
    • 如果仓库中没有产品 , 则生产者将产品放入仓库 , 否则停止生产并等待 , 直到仓库中的产品被消费者取走为止
    • 如果仓库中放有产品 , 则消费者可以将产品取走消费 , 否则停止消费并等待 ,直到仓库中再次放入产品为止

多线程 - 图30

  • 线程通信-分析
    这是一个线程同步问题 , 生产者和消费者共享同一个资源 , 并且生产者和消费者之间相互依赖 , 互为条件。
    • 对于生产者 , 没有生产产品之前 , 要通知消费者等待。而生产了产品之后 , 又需要马上通知消费者消费
    • 对于消费者 , 在消费之后 , 要通知生产者已经结束消费 , 需要生产新的产品以供消费.
    • 在生产者消费者问题中 , 仅有synchronized是不够的
      • synchronized 可阻止并发更新同一个共享资源 , 实现了同步
      • synchronized 不能用来实现不同线程之间的消息传递 (通信)

1)线程通信

  • Java提供了几个方法解决线程之间的通信问题
    • wait() :表示线程一直等待,直到其他线程通知。与sleep不同,会释放锁。
    • wait(long timeout) :指定等待的毫秒数
    • notify() :唤醒处于等待状态的线程中优先级最高的一个线程结束等待
    • notifyAll() :唤醒处于等待状态的所有线程结束等待,优先级别高的线程优先调度
  • 这三个方法只有在synchronized方法或synchronized代码块中才能使用,否则会报java.lang.IllegalMonitorStateException异常。
  • 均是Object类的方法 , 都只能在同步方法或者同步代码块中使用,否则会抛出异常IllegalMonitorStateException

wait() 方法

  • 在当前线程中调用方法:对象名.wait()
  • 使当前线程进入等待状态,直到另一线程对该对象发出notify (或notifyAll) 为止。
  • 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)
  • 调用此方法后,当前线程将释放对象监控权,然后进入等待
  • 在当前线程被notify后,要重新获得监控权,然后从断点处继续代码的执行。

notify()/notifyAll()

  • 在当前线程中调用方法:对象名.notify()
  • 功能:唤醒等待该对象监控权的一个/所有线程。
  • 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)

2)解决方式1

并发协作模型 “ 生产者 / 消费者模式 ” —-> 管程法

  • 生产者 : 负责生产数据的模块 (可能是方法 , 对象 , 线程 , 进程) ;
  • 消费者 : 负责处理数据的模块 (可能是方法 , 对象 , 线程 , 进程) ;
  • 缓冲区 : 消费者不能直接使用生产者的数据 , 他们之间有个 “ 缓冲区

生产者将生产好的数据放入缓冲区 , 消费者从缓冲区拿出数据
多线程 - 图31

案例演示:

  1. package com.wang.thread;
  2. //生产者消费者模型 --> 利用缓冲区解决:管程法
  3. //生产者,消费者,产品,缓冲区
  4. public class PCTest {
  5. public static void main(String[] args) {
  6. SynContainer container = new SynContainer();
  7. new Productor(container).start();
  8. new Consumer(container).start();
  9. }
  10. }
  11. //生产者
  12. class Productor extends Thread {
  13. SynContainer container;
  14. public Productor(SynContainer container) {
  15. this.container = container;
  16. }
  17. //生产
  18. @Override
  19. public void run() {
  20. for (int i = 0; i < 100; i++) {
  21. container.push(new Chicken(i));
  22. System.out.println("生产了" + i + "只鸡");
  23. }
  24. }
  25. }
  26. //消费者
  27. class Consumer extends Thread {
  28. SynContainer container;
  29. public Consumer(SynContainer container) {
  30. this.container = container;
  31. }
  32. //消费
  33. @Override
  34. public void run() {
  35. for (int i = 0; i < 100; i++) {
  36. System.out.println("消费了-->" + container.pop().id + "只鸡");
  37. }
  38. }
  39. }
  40. //产品
  41. class Chicken {
  42. int id;//产品编号
  43. public Chicken(int id) {
  44. this.id = id;
  45. }
  46. }
  47. //缓冲区
  48. class SynContainer {
  49. //需要一个容器大小
  50. Chicken[] chickens = new Chicken[10];
  51. //容器计数器
  52. int count = 0;
  53. //生产者放入产品
  54. public synchronized void push(Chicken chicken) {
  55. //如果容器满了,就需要等待消费者消费
  56. if (count == chickens.length) {
  57. //通知消费者消费,生产等待
  58. try {
  59. this.wait();
  60. } catch (InterruptedException e) {
  61. e.printStackTrace();
  62. }
  63. }
  64. //如果没有满,就需要丢入产品
  65. chickens[count] = chicken;
  66. count++;
  67. //可以通知消费者消费
  68. this.notifyAll();
  69. }
  70. //消费者消费产品
  71. public synchronized Chicken pop() {
  72. //判断能否消费
  73. if (count == 0) {
  74. //等待生产者生产,消费者等待
  75. try {
  76. this.wait();
  77. } catch (InterruptedException e) {
  78. e.printStackTrace();
  79. }
  80. }
  81. //如果可以消费
  82. count--;
  83. Chicken chicken = chickens[count];
  84. //通知生产者生产
  85. this.notifyAll();
  86. return chicken;
  87. }
  88. }

3)解决方式2

并发协作模型 “ 生产者 / 消费者模式 ” —-> 信号灯法

  1. package com.wang.thread;
  2. //生产者消费者问题2:信号灯法,标志位解决
  3. public class PCTest2 {
  4. public static void main(String[] args) {
  5. TV tv = new TV();
  6. new Player(tv).start();
  7. new Watcher(tv).start();
  8. }
  9. }
  10. //生产者 --> 演员
  11. class Player extends Thread {
  12. TV tv;
  13. public Player(TV tv){
  14. this.tv=tv;
  15. }
  16. @Override
  17. public void run() {
  18. for (int i = 0; i < 20; i++) {
  19. if (i%2==0){
  20. this.tv.play("快乐大本营播放中");
  21. }else {
  22. this.tv.play("抖音:记录美好生活");
  23. }
  24. }
  25. }
  26. }
  27. //消费者 --> 观众
  28. class Watcher extends Thread {
  29. TV tv;
  30. public Watcher(TV tv){
  31. this.tv=tv;
  32. }
  33. @Override
  34. public void run() {
  35. for (int i = 0; i < 20; i++) {
  36. tv.watch();
  37. }
  38. }
  39. }
  40. //产品 --> 节目
  41. class TV {
  42. //演员表演,观众等待 T
  43. //观众观看,演员等待 F
  44. String voice;
  45. boolean flag = true;
  46. //表演
  47. public synchronized void play(String voice) {
  48. if (!flag){
  49. try {
  50. this.wait();
  51. } catch (InterruptedException e) {
  52. e.printStackTrace();
  53. }
  54. }
  55. System.out.println("演员表演了:" + voice);
  56. //通知观众观看
  57. this.notifyAll();//通知唤醒
  58. this.voice=voice;
  59. this.flag= !this.flag;
  60. }
  61. //观看
  62. public synchronized void watch( ) {
  63. if (flag) {
  64. try {
  65. this.wait();
  66. } catch (InterruptedException e) {
  67. e.printStackTrace();
  68. }
  69. }
  70. System.out.println("观众观看了:" + voice);
  71. //通知演员表演
  72. this.notifyAll();//通知唤醒
  73. this.voice = voice;
  74. this.flag = !this.flag;
  75. }
  76. }

JDK5.0新增线程创建方式

1)新增方式一:实现Callable接口

  • 与使用Runnable相比,Callable功能更强大些
    • 相比run()方法,可以有返回值
    • 方法可以抛出异常
    • 支持泛型的返回值
    • 需要借助FutureTask类,比如获取返回结果
  • Future接口
    • 可以对具体Runnable、Callable任务的执行结果进行取消、查询是否完成、获取结果等。
    • FutrueTask是Futrue接口的唯一的实现类
    • FutureTask 同时实现了Runnable, Future接口。它既可以作为Runnable被线程执行,又可以作为Future得到Callable的返回值

2)新增方式二:使用线程池

  • 背景:经常创建和销毁、使用量特别大的资源,比如并发情况下的线程,对性能影响很大。
  • 思路:提前创建好多个线程,放入线程池中,使用时直接获取,使用完放回池中。可以避免频繁创建销毁、实现重复利用。
  • 好处:
    • 提高响应速度(减少了创建新线程的时间)
    • 降低资源消耗(重复利用线程池中线程,不需要每次都创建)
    • 便于线程管理
      • corePoolSize:核心池的大小
      • maximumPoolSize:最大线程数
      • keepAliveTime:线程没有任务时最多保持多长时间后会终止
  • 线程池相关API
    • JDK 5.0起提供了线程池相关API:ExecutorServiceExecutors
    • ExecutorService:真正的线程池接口。常见子类ThreadPoolExecutor
      • void execute(Runnable command) :执行任务/命令,没有返回值,一般用来执行Runnable
      • Future submit(Callable task):执行任务,有返回值,一般又来执行Callable
      • void shutdown() :关闭连接池
    • Executors:工具类、线程池的工厂类,用于创建并返回不同类型的线程池。
      • Executors.newCachedThreadPool():创建一个可根据需要创建新线程的线程池
      • Executors.newFixedThreadPool(int n):创建一个可重用固定线程数的线程池,参数n为池子大小
      • Executors.newSingleThreadExecutor() :创建一个只有一个线程的线程池
      • Executors.newScheduledThreadPool(n):创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。
  1. package com.wang.thread;
  2. import java.util.concurrent.ExecutorService;
  3. import java.util.concurrent.Executors;
  4. //线程池
  5. public class PoolTest {
  6. public static void main(String[] args) {
  7. //1.创建服务,创建线程池
  8. //newFixedThreadPool(int n) 参数n为线程池的大小
  9. ExecutorService service = Executors.newFixedThreadPool(10);
  10. //2.执行
  11. service.execute(new MyThread());
  12. service.execute(new MyThread());
  13. service.execute(new MyThread());
  14. service.execute(new MyThread());
  15. //3.关闭线程池
  16. service.shutdown();
  17. }
  18. }
  19. class MyThread implements Runnable {
  20. @Override
  21. public void run() {
  22. System.out.println(Thread.currentThread().getName());
  23. }
  24. }

多线程 - 图32

回顾总结

回顾总结线程的创建

  1. package com.wang.thread;
  2. import java.util.concurrent.Callable;
  3. import java.util.concurrent.ExecutionException;
  4. import java.util.concurrent.FutureTask;
  5. //回顾总结线程的创建
  6. public class ThreadNew {
  7. public static void main(String[] args) {
  8. //方式1线程启动
  9. new MyThread1().start();
  10. //方式2线程启动
  11. new Thread(new MyThread2()).start();
  12. //方式3线程启动
  13. FutureTask<Integer> futureTask = new FutureTask<>(new MyThread3());
  14. new Thread(futureTask).start();
  15. //打印返回值
  16. try {
  17. Integer integer = futureTask.get();
  18. System.out.println(integer);
  19. } catch (InterruptedException e) {
  20. e.printStackTrace();
  21. } catch (ExecutionException e) {
  22. e.printStackTrace();
  23. }
  24. }
  25. }
  26. //1.继承Thread类
  27. class MyThread1 extends Thread {
  28. @Override
  29. public void run() {
  30. System.out.println("MyThread1");
  31. }
  32. }
  33. //2.实现Runnable接口
  34. class MyThread2 implements Runnable {
  35. @Override
  36. public void run() {
  37. System.out.println("MyThread2");
  38. }
  39. }
  40. //3.实现Callable接口
  41. class MyThread3 implements Callable<Integer> {
  42. @Override
  43. public Integer call() throws Exception {
  44. System.out.println("MyThread3");
  45. return 100;
  46. }
  47. }

多线程 - 图33