关于Java的Condition接口最佳理解方式

 更新时间:2023年05月09日 08:59:17   作者:JavaEdge.  
这篇文章主要介绍了关于Java的Condition接口最佳理解方式,Condition就是实现了管程里面的条件变量,Java 语言内置的管程里只有一个条件变量,而Lock&Condition实现的管程支持多个条件变量,需要的朋友可以参考下

Condition就是实现了管程里面的条件变量。

Java 语言内置的管程里只有一个条件变量,而Lock&Condition实现的管程支持多个条件变量。 支持多个条件变量,能让代码可读性更好,实现也更容易。

例如,你看我这里实现一个阻塞队列,就需要两个条件变量:

  • 队列不空 空队列自然没有元素能出队队列不满
  • 队列已满,当然也不可有元素再入队

Lock和Condition实现的管程,线程等待和通知需要调用await()、signal()、signalAll(),它们的语义和wait()、notify()、notifyAll()相同。

  • Lock&Condition实现的管程里只能使用await()、signal()、signalAll()
  • synchronized实现的管程才能使用wait()、notify()、notifyAll()

如果在Lock&Condition实现的管程里调用wait()、notify()、notifyAll(),你距离离职就更近一步了。

Thread.sleep() V.S Condition.await()

Object.wait()和Condition.await()的原理是基本一致的,不同在于Condition.await()底层是调用LockSupport.park()实现阻塞当前线程。它在阻塞当前线程前,其实还做了:

  1. 把当前线程添加到条件队列
  2. 完全释放锁,即让state=0,然后才调用LockSupport.park()阻塞当前线程

JDK的Lock和Condition不过就是管程的一种实现,一般如何使用呢?

什么是同步与异步?

  • 同步 调用方需要等待结果
  • 异步 不需要等待结果

代码里如何实现异步?

  • 调用方创建一个子线程,在子线程中执行方法调用,即异步调用
  • 方法实现时,创建一个新的线程执行主要逻辑,主线程直接return,即异步方法。

异步场景挺多,比如TCP协议本身是异步的,日常的RPC调用,在TCP协议层面,发送完RPC请求后,线程不会等待RPC响应结果。

是不是好奇了,明明日常使用的RPC调用都是同步的呀?这到底是同步还是异步?

这肯定有人帮忙实现了异步转同步。比如RPC框架Dubbo,具体它是怎么做到的呢?

对于下面一个简单的RPC调用,默认情况下sayHello()是个同步方法,即执行service.sayHello(“dubbo”)时,线程会停下来等结果。

DemoService service = 初始化部分省略
String message = service.sayHello("dubbo");
System.out.println(message);

若此时dump调用线程的调用栈

发现调用线程阻塞了,线程状态是TIMED_WAITING。本来发送请求是异步的,但是调用线程却阻塞了,说明Dubbo帮我们做了异步转同步的事情。通过调用栈看到线程是阻塞在DefaultFuture.get(),所以Dubbo异步转同步的功能应该是通过DefaultFuture实现。

DefaultFuture.get()之前发生了什么呢:

我们的期望:

RPC返回结果前,阻塞调用线程,让调用线程等待
RPC返回结果后,唤醒调用线程,让调用线程重新执行

这就是经典的等待-通知机制,即管程的实现方案。

  • 看看Dubbo是怎么实现的。

到此这篇关于关于Java的Condition接口最佳理解方式的文章就介绍到这了,更多相关Java的Condition接口内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java中List转Map的几种具体实现方式和特点

    Java中List转Map的几种具体实现方式和特点

    这篇文章主要介绍了几种常用的List转Map的方式,包括使用for循环遍历、Java8StreamAPI、ApacheCommonsCollections和GoogleGuava,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2025-01-01
  • Spring Boot集成LangChain来实现Rag应用的问题小结

    Spring Boot集成LangChain来实现Rag应用的问题小结

    检索增强生成(RAG)是一种优化大型语言模型(LLM)输出的技术,通过引用权威知识库以增强模型的准确性和相关性,RAG允许LLM在不重新训练的情况下访问特定领域的知识,提高了其在各种应用中的实用性和信任度,感兴趣的朋友跟随小编一起看看吧
    2024-09-09
  • 基于swing开发弹幕播放器

    基于swing开发弹幕播放器

    这篇文章主要为大家详细介绍了基于swing实现弹幕播放器的开发过程,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • spring状态机模式使用小结

    spring状态机模式使用小结

    说起Spring状态机,大家很容易联想到这个状态机和设计模式中状态模式的区别是啥呢?没错,Spring状态机就是状态模式的一种实现,在介绍Spring状态机之前,让我们来看看设计模式中的状态模式,需要的朋友可以参考下
    2024-04-04
  • Springboot No bean named 'XXXXX' available 问题解决方法

    Springboot No bean named 'XXXXX' available 问

    这篇文章主要介绍了Springboot No bean named 'XXXXX' available 问题解决方法,解决方法也很简单,尽量规范类的命名,注解中指定bean名称,本文给大家介绍的非常详细,需要的朋友可以参考下
    2023-07-07
  • Java日常练习题,每天进步一点点(45)

    Java日常练习题,每天进步一点点(45)

    下面小编就为大家带来一篇Java基础的几道练习题(分享)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧,希望可以帮到你
    2021-07-07
  • Java中的final关键字使用方式

    Java中的final关键字使用方式

    这篇文章主要介绍了Java中的final关键字使用方式,final 关键字用于修饰不可改变内容,更多相关梳理总结,需要的小伙伴可以参考下面文章内容
    2022-06-06
  • SpringBoot通过请求对象获取输入流无数据

    SpringBoot通过请求对象获取输入流无数据

    这篇文章主要介绍了使用SpringBoot通过请求对象获取输入流无数据,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • Java实现平滑加权轮询算法之降权和提权详解

    Java实现平滑加权轮询算法之降权和提权详解

    所有负载均衡的场景几乎都会用到这个平滑加权轮询算法,下面这篇文章主要给大家介绍了关于Java实现平滑加权轮询算法之降权和提权的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2022-04-04
  • java中下拉框select和单选按钮的回显操作

    java中下拉框select和单选按钮的回显操作

    这篇文章主要介绍了java中下拉框select和单选按钮的回显操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10

最新评论