看了网上非常多的运行代码,很多都是重复的再说一件事,可能对于java老鸟来说,理解java的多线程是非常容易的事情,但是对于我这样的菜鸟来说,这个实在有点难,可能是我太菜了,网上重复的陈述对于我理解这个问题一点帮助都没有.所以这里我写下我对于这个问题的理解,目的是为了防止我忘记.

还是从代码实例开始讲起:

  1. import java.util.Arrays;
  2. import java.util.LinkedList;
  3. import java.util.List;
  4. import java.util.concurrent.locks.Condition;
  5. import java.util.concurrent.locks.ReentrantLock;
  6. import java.util.function.Predicate;
  7.  
  8. public class Main {
  9. public static void main(String[] args) throws InterruptedException {
  10. MyBlockingQueue<Integer> queue = new MyBlockingQueue<>(1);
  11. for (int i = 0; i < 10; i++) {
  12. int data = i;
  13. new Thread(() -> {
  14. try {
  15. queue.enqueue(data);
  16. } catch (InterruptedException e) {
  17. e.printStackTrace();
  18. }
  19. }).start();
  20. }
  21. System.out.println("1111111");
  22. for(int i=0;i<10;i++){
  23. new Thread(() -> {
  24. try {
  25. queue.dequeue();
  26. }catch (InterruptedException e){
  27. e.printStackTrace();
  28. }
  29. }).start();
  30. }
  31. }
  32. public static class MyBlockingQueue<E> {
  33. int size;//阻塞队列最大容量
  34. ReentrantLock lock = new ReentrantLock(true);
  35. LinkedList<E> list=new LinkedList<>();//队列底层实现
  36. Condition notFull = lock.newCondition();//队列满时的等待条件
  37. Condition notEmpty = lock.newCondition();//队列空时的等待条件
  38. public MyBlockingQueue(int size) {
  39. this.size = size;
  40. }
  41. public void enqueue(E e) throws InterruptedException {
  42. lock.lock();
  43. try {
  44. while(list.size() ==size)//队列已满,在notFull条件上等待
  45. notFull.await();
  46.  
  47. list.add(e);//入队:加入链表末尾
  48. System.out.println("入队:" +e);
  49. notEmpty.signal(); //通知在notEmpty条件上等待的线程
  50. } finally {
  51. lock.unlock();
  52. }
  53. }
  54. public E dequeue() throws InterruptedException {
  55. E e;
  56. lock.lock();
  57. try {
  58. while(list.size() == 0)
  59. notEmpty.await();
  60. e = list.removeFirst();//出队:移除链表首元素
  61. System.out.println("出队:"+e);
  62. notFull.signal();//通知在notFull条件上等待的线程
  63. return e;
  64. } finally {
  65. lock.unlock();
  66. }
  67. }
  68. }
  69. }

代码来自菜鸟教程

主函数启动了20个线程,前10个是入队的后10个是出队的,我们可以看啊可能输出结果,

  1. new Thread(() -> {
    try {
    queue.enqueue(data);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }
    }).start(); 注意到线程实现,这个是lambda表达式实现Runable接口.
  1. 入队:0
  2. 1111111
  3. 出队:0
  4. 入队:2
  5. 出队:2
  6. 入队:1
  7. 出队:1
  8. 入队:3
  9. 出队:3
  10. 入队:4
  11. 出队:4
  12. 入队:5
  13. 出队:5
  14. 入队:6
  15. 出队:6
  16. 入队:7
  17. 出队:7
  18. 入队:8
  19. 出队:8
  20. 入队:9
  21. 出队:9

可以看到1111111在第一个出队之前,队列容量为1,也就是说头10个入队进程只有第一个成功了,其他均被阻塞.

并且出队入队顺序是按照循环顺序的,说明锁是按照请求顺序来获取的,先到先得,这个说的就是公平锁的意思,其实ReentrantLock既可以是公平锁也可以是非公平锁,其初始化的时候,往构造函数里面传入true则为公平锁,false则为非公平锁.

至于什么是可重入锁,可以看看这篇博客https://blog.csdn.net/qq_38737992/article/details/90613821,这个是可重入锁的实例.

其中有一行代码很奇怪,lock.lock();对于我这样的萌新很好奇这个一行代码到底发生了什么,网上很多都是说获得锁,但是"获得"这个实在难以太不具体,所以我自己想象了一下,感觉大致上就是这样的一张图:

结合我粗浅的经验猜测:jvm只有一个就绪队列,就绪队列里面的线程按照队列顺序使用cpu资源,若不加锁,那么所有线程都可以按序取得资源,但是由于面向对象了,所以不好直接控制就绪队列里面线程的入队顺序,这个时候就需要加锁来控制线程的运行顺序来保证处理逻辑正确.(其实也不不是那么严格的队列,就绪状态的线程如果是非公平锁一般会随机先后的运行,说是队列而已,其实就是表达就绪状态)

结合代码的lock()方法,如果有线程进入cpu并且调用lock(),如果该锁没有被其他线程获取过,那么这个线程可以使用cpu时间,如果该锁已经被其他线程获取了,那么该线程会给阻塞,进入阻塞队列, 这样来说的话,其实"获取"这个词也没什么难以理解的,其实就是一个标记而已,然后lock()方法其实就只是判断当前线程是使用cpu时间,还是进入阻塞队列而已..

在看看unlock()方法,由上面的图帮助,其实这个也很好理解,其实就是把阻塞队列的队首的线程出队,然后进入就绪队列而已.

可以猜测,如果运行过程中有多个锁实例,那么就会有多少个可能阻塞的线程,那么除了使用用多个锁,其实还有别的方法来增加阻塞线程,就是使用Condition类,需要指出的是condition类的await()方法,会阻塞当前线程,然后自动解除当前线程获取的锁(这点尤其重要),切换线程,如果其他线程中有唤醒,那么这个在被唤醒后线程会从await()的位置继续往下运行,所以一般要配合while循环使用,如果某线程被唤醒,那么它对于它之前获取的锁,也将重新获取,如果此时该锁已经被另外一个线程获取,且还没有解锁,此时的唤醒就会出错,会出现莫名其妙的错误,所以需要设置一个volatile变量来检测线程的运行状态,所以await()方法前后都要检测.

这里提出一道题,来自leetcode,要求使用condition类来写,

题意:

编写一个可以从 1 到 n 输出代表这个数字的字符串的程序,但是:

如果这个数字可以被 3 整除,输出 "fizz"。
如果这个数字可以被 5 整除,输出 "buzz"。
如果这个数字可以同时被 3 和 5 整除,输出 "fizzbuzz"。
例如,当 n = 15,输出: 1, 2, fizz, 4, buzz, fizz, 7, 8, fizz, buzz, 11, fizz, 13, 14, fizzbuzz。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/fizz-buzz-multithreaded

解法:

  1. import java.util.Random;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.Lock;
  5. import java.util.concurrent.locks.ReentrantLock;
  6.  
  7. public class Main {
  8. static void printFizz(int x){
  9. System.out.printf("%d:Fizz,\n",x);
  10. }
  11. static void printBuzz(int x){
  12. System.out.printf("%d:Buzz,\n",x);
  13. }
  14. static void printFizzBuzz(int x){
  15. System.out.printf("%d:FizzBuzz,\n",x);
  16. }
  17. static void printaccpt(int x){
  18. System.out.printf("%d,\n",x);
  19. }
  20. static volatile int now=1;
  21. static ReentrantLock lock=new ReentrantLock();
  22. static Condition k1=lock.newCondition();
  23. public static void test(int n) {
  24.  
  25. new Thread(()->{
  26. while(now<=n){
  27. lock.lock();
  28. try{
  29. while(now%5==0||now%3!=0){
  30. if(now>n) throw new InterruptedException();
  31. k1.await();
  32. if(now>n) throw new InterruptedException();
  33. }
  34. printFizz(now);
  35. now++;
  36. k1.signalAll();
  37. } catch (InterruptedException e) {
  38. break;
  39. //e.printStackTrace();
  40. } finally{
  41. lock.unlock();
  42. }
  43. }
  44. System.out.println("Thread 1 is over");
  45. }).start();
  46.  
  47. new Thread(()->{
  48. while(now<=n){
  49. lock.lock();
  50. try{
  51.  
  52. while(now%5!=0||now%3==0) {
  53. if(now>n) throw new InterruptedException();
  54. k1.await();
  55. if(now>n) throw new InterruptedException();
  56. }
  57. printBuzz(now);
  58. now++;
  59. k1.signalAll();
  60. } catch (InterruptedException e) {
  61. break;
  62. // e.printStackTrace();
  63. } finally{
  64. lock.unlock();
  65. }
  66. }
  67. System.out.println("Thread 2 is over");
  68. }).start();
  69.  
  70. new Thread(()->{
  71. while(now<=n){
  72. lock.lock();
  73. try{
  74. while(now%5!=0||now%3!=0) {
  75. if(now>n) throw new InterruptedException();
  76. k1.await();
  77. if(now>n) throw new InterruptedException();
  78. }
  79. printFizzBuzz(now);
  80. now++;
  81. k1.signalAll();
  82. } catch (InterruptedException e) {
  83. break;
  84. //Thread.interrupted();
  85. //e.printStackTrace();
  86. } finally{
  87. lock.unlock();
  88. }
  89. }
  90. System.out.println("Thread 3 is over");
  91. }).start();
  92.  
  93. new Thread(()->{
  94. while(now<=n){
  95. lock.lock();
  96. try{
  97. while(now%5==0||now%3==0) {
  98. if(now>n) throw new InterruptedException();
  99. k1.await();
  100. if(now>n) throw new InterruptedException();
  101. }
  102. printaccpt(now);
  103. now++;
  104. k1.signalAll();
  105. }catch (InterruptedException e){
  106. break;
  107. //Thread.interrupted();
  108. //e.printStackTrace();
  109. }
  110. finally{
  111. lock.unlock();
  112. }
  113. }
  114. System.out.println("Thread 4 is over");
  115. }).start();
  116. }
  117.  
  118. public static void main(String[] args) throws InterruptedException {
  119. test(30);
  120. }
  121. }

java多线程,多线程加锁以及Condition类的使用的更多相关文章

  1. Java:多线程,使用同步锁(Lock)时利用Condition类实现线程间通信

    如果程序不使用synchronized关键字来保证同步,而是直接使用Lock对象来保证同步,则系统中不存在隐式的同步监视器,也就不能用wait().notify().notifyAll()方法进行线程 ...

  2. java 多线程 Thread 锁ReentrantLock;Condition等待与通知;公平锁

    1,介绍: import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock;   在JA ...

  3. java核心-多线程-Java多线程编程涉及到包、类

    Java有关多线程编程设计的类主要涉及两个包java.lang和java.util.concurrent两个包 java.lang包,主要是线程基础类 <1>Thread <2> ...

  4. Java并发多线程 - 并发工具类JUC

    安全共享对象策略 1.线程限制 : 一个被线程限制的对象,由线程独占,并且只能被占有它的线程修改 2.共享只读 : 一个共享只读的对象,在没有额外同步的情况下,可以被多个线程并发访问, 但是任何线程都 ...

  5. java多线程无锁和工具类

    1 无锁 (1) cas (compare and swap) 设置值的时候,会比较当前值和当时拿到的值是否相同,如果相同则设值,不同则拿新值重复过程:注意,在设置值的时候,取值+比较+设值 是一条c ...

  6. java并发多线程显式锁Condition条件简介分析与监视器 多线程下篇(四)

    Lock接口提供了方法Condition newCondition();用于获取对应锁的条件,可以在这个条件对象上调用监视器方法 可以理解为,原本借助于synchronized关键字以及锁对象,配备了 ...

  7. java多线程并发去调用一个类的静态方法安全性探讨

    java多线程并发去调用一个类的静态方法安全性探讨 转自:http://blog.csdn.net/weibin_6388/article/details/50750035   这篇文章主要讲多线程对 ...

  8. 2019.12.12 Java的多线程&匿名类

    Java基础(深入了解概念为主) 匿名类 定义 Java匿名类很像局部或内联系,只是没有明细.我们可以利用匿名类,同时定义并实例化一个类.只有局部类仅被使用一次时才应该这么做. 匿名类不能有显式定义的 ...

  9. 【Java多线程系列五】列表类

    一些列表类及其特性  类 线程安全 Iterator 特性 说明 Vector 是 fail-fast 内部方法用synchronized修饰,因此执行效率较低 1. 线程安全的列表类并不意味着调用它 ...

随机推荐

  1. 安装配置 Android Studio

    概述 Android Studio 本身应该是开箱即用的,但是由于 dl.google.com 访问太慢,导致了这样那样的问题,因此我们只需要改一下 hosts 就行了 具体步骤 在Ping检测网站查 ...

  2. 使用 App Inventor 2 开发简单的安卓小游戏

    App Inventor2 是一个简单的在线开发安卓应用程序的工具,通过此工具,我们可以很轻松地开发安卓应用. 这里介绍的是笔者自己写的一个小游戏,游戏中玩家通过左右倾斜手机控制“水库”的左右移动,收 ...

  3. C#事件浅淡(1)

    最近在写C#,感觉事件这个机制很好,可是怎么实现自己定义的事件呢?查了资料有的不全有的不完整,有的太深,自己写一个简单的例子. 原则 1,定义一个事件信息类(标准的都继承EventArgs) 2.定义 ...

  4. Mybatis常见配置错误总结

    Mybatis常见配置错误总结 <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionF ...

  5. WWW网络请求

    采用WWW获取网络数据: (一)get 1)天气数据下载 private string weatherApi = "http://www.sojson.com/open/api/weathe ...

  6. SpringCloud之Nacos服务发现(十七)

    一 Nacos简介 Nacos是以服务为主要服务对象的中间件,Nacos支持所有主流的服务发现.配置和管理. Nacos主要提供以下四大功能: 服务发现与服务健康检查 Nacos使服务更容易注册自己并 ...

  7. Java中的substring()用法

    String str = "Hello Java World!"; Method1:  substring(int beginIndex) 返回从起始位置(beginIndex)至 ...

  8. 过滤器、拦截器和AOP的分析与对比

    目录 一.过滤器(Filter) 1.1 简介 1.2 应用场景 1.3 源码分析 二.拦截器(Interceptor) 2.1 简介 2.2 应用场景 2.2 源码分析 三.面向切面编程(AOP) ...

  9. 在移动硬盘中安装win10和macos双系统

    本文通过在SSD移动硬盘中安装win10和macos双系统,实现操作系统随身携带 小慢哥的原创文章,欢迎转载 目录 ▪ 目标 ▪ 准备工作 ▪ Step1. 清空分区,转换为GPT ▪ Step2. ...

  10. Python3爬虫(1)_使用Urllib进行网络爬取

    网络爬虫 又被称为网页蜘蛛,网络机器人,在FOAF社区中间,更经常的称为网页追逐者,是一种按照一定的规则,自动地抓取万维网信息的程序或者脚本.另外一些不常使用的名字还有蚂蚁.自动索引.模拟程序或者蠕虫 ...