java使用wait()和notify()线程间通讯的实现

 更新时间:2023年09月01日 15:54:25   作者:L-960  
Java 线程通信是将多个独立的线程个体进行关联处理,使得线程与线程之间能进行相互通信,本文就介绍了java使用wait()和notify()线程间通讯的实现,感兴趣的可以了解一下

线程.wait()

当一个线程调用 线程.wait()时,它会释放对象的锁,让其他线程可以获得这个锁并执行相应的同步代码块,同时,如果未配置超时时间,该线程会无限期等待,直到接收到线程.notify()信号,注意,一般情况下要在同步代码块中执行。

线程.notify()

当一个线程调用 线程.notify()时,它会给线程发送一个信号,让其在线程.notify()卡住的地方继续执行,注意,一般情况下要在同步代码块中执行。

示例

下面是一个演示线程间通讯的示例,在一个定时任务中达成某个条件时,通知主线程继续执行,同时关闭该定时任务:

    public static void main(String[] args) throws InterruptedException {
        final Thread main = Thread.currentThread();
        main.setName("main thread");
        // 创建一个默认定时任务
        ScheduledThreadPoolExecutor executor = new ScheduledThreadPoolExecutor(
                5,  // 核心线程数
                new ThreadPoolExecutor.CallerRunsPolicy()  // 队列满后的拒绝策略
        );
        AtomicInteger flag = new AtomicInteger(0);
        // 执行定时任务
        final ScheduledFuture<?> queryTask = executor.scheduleAtFixedRate(() -> {
            // 在这里执行您的定时任务逻辑
            final int i = flag.addAndGet(1);
            log.info("定时任务执行了1!flag={}", i);
            if (i == 5) {
                synchronized (main) {
                    System.out.println("通知主线程取消该任务");
                    main.notify();
                }
            }
        }, 0, 3, TimeUnit.SECONDS);
        final ScheduledFuture<?> printTask = executor.scheduleAtFixedRate(() -> {
            // 在这里执行您的定时任务逻辑
            log.info("定时任务执行了2!");
        }, 0, 3, TimeUnit.SECONDS);
        log.info("main wait");
        synchronized (main) {
            /**
             *
             主线程在执行 main.wait() 进入等待状态后,确实会释放 main 对象的锁,而不是一直占用锁。
             这是 Java 中 wait() 方法的特性之一。
             当一个线程调用 wait() 方法时,它会释放对象的锁,让其他线程可以获得这个锁并执行相应的同步代码块。
             */
            main.wait();
        }
        log.info("main 活了");
        // 取消定时任务
        queryTask.cancel(true);
        log.info("scheduledFuture被取消");
    }

运行结果:

11:14:50.838 [main thread] INFO MyTaskConfig - main wait
11:14:50.838 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了2!
11:14:50.838 [pool-1-thread-1] INFO MyTaskConfig - 定时任务执行了1!flag=1
11:14:53.845 [pool-1-thread-1] INFO MyTaskConfig - 定时任务执行了2!
11:14:53.845 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了1!flag=2
11:14:56.841 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了1!flag=3
11:14:56.842 [pool-1-thread-3] INFO MyTaskConfig - 定时任务执行了2!
11:14:59.841 [pool-1-thread-4] INFO MyTaskConfig - 定时任务执行了1!flag=4
11:14:59.841 [pool-1-thread-1] INFO MyTaskConfig - 定时任务执行了2!
11:15:02.848 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了2!
11:15:02.848 [pool-1-thread-4] INFO MyTaskConfig - 定时任务执行了1!flag=5
通知主线程取消该任务
11:15:02.848 [main thread] INFO MyTaskConfig - main 活了
11:15:02.848 [main thread] INFO MyTaskConfig - scheduledFuture被取消
11:15:05.835 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了2!
11:15:08.843 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了2!
11:15:11.836 [pool-1-thread-2] INFO MyTaskConfig - 定时任务执行了2!

到此这篇关于java使用wait()和notify()线程间通讯的实现的文章就介绍到这了,更多相关java wait()和notify()线程间通讯内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 一次mybatis连接查询遇到的坑实战记录

    一次mybatis连接查询遇到的坑实战记录

    这篇文章主要给大家介绍了关于一次mybatis连接查询遇到的坑的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • 序列化版本号serialVersionUID的作用_动力节点Java学院整理

    序列化版本号serialVersionUID的作用_动力节点Java学院整理

    Java序列化是将一个对象编码成一个字节流,反序列化将字节流编码转换成一个对象,这篇文章主要介绍了序列化版本号serialVersionUID的作用,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05
  • Java数据结构贪心算法的实现

    Java数据结构贪心算法的实现

    本文主要介绍了Java数据结构贪心算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2007-03-03
  • SpringCloud Feign多参数传递及需要注意的问题

    SpringCloud Feign多参数传递及需要注意的问题

    这篇文章主要介绍了SpringCloud Feign多参数传递及需要注意的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • Java中token的存储和获取实例代码

    Java中token的存储和获取实例代码

    关于java获取微信Token验证的问题相信很多人都遇见过,尤其是对刚接触微信开发的人来说确实有点棘手,下面这篇文章主要给大家介绍了关于Java中token存储和获取的相关资料,需要的朋友可以参考下
    2022-08-08
  • 详解Spring Cloud Zuul网关修改为短连接方法

    详解Spring Cloud Zuul网关修改为短连接方法

    本文主要介绍了详解Spring Cloud Zuul网关修改为短连接方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-04-04
  • 到底如何设置Java线程池的大小的方法示例

    到底如何设置Java线程池的大小的方法示例

    在我们日常业务开发过程中,或多或少都会用到并发的功能。那么并发线程池到底设置多大呢?文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09
  • JDBC使用游标实现分页查询的方法

    JDBC使用游标实现分页查询的方法

    这篇文章主要介绍了JDBC使用游标实现分页查询的方法,实例分析了jdbc查询过程中游标的使用及查询分页相关实现技巧,需要的朋友可以参考下
    2016-08-08
  • Spring循环依赖实现过程揭秘

    Spring循环依赖实现过程揭秘

    这篇文章主要介绍了Spring循环依赖实现过程,Spring的解决循环依赖是有前置条件的,要解决循环依赖我们首先要了解Spring Bean对象的创建过程和依赖注入的方式
    2023-01-01
  • Java的非阻塞队列ConcurrentLinkedQueue解读

    Java的非阻塞队列ConcurrentLinkedQueue解读

    这篇文章主要介绍了Java的非阻塞队列ConcurrentLinkedQueue解读,在并发编程中,有时候需要使用线程安全的队列,如果要实现一个线程安全的队列有两种方式:一种是使用阻塞算法,另一种是使用非阻塞算法,需要的朋友可以参考下
    2023-12-12

最新评论