java多线程 线程交替执行(同步)的各种实现方案

目录

  • java多线程 线程交替执行(同步)的各种实现方案
    • 需求
  • 指定具体执行顺序
    • 实现一:wait_notify机制 + Thread标志位
    • 实现二:lock_condition机制 + Thread标志位
    • 实现三:semaphore信号量
  • 不指定具体执行顺序,只交替执行
    • 实现一:synchronized和wait、notify实现
    • 实现二:lock和condition实现
  • 总结

java多线程 线程交替执行(同步)的各种实现方案

需求

有两个线程,A线程内容是输出数字1到26,B线程内容是输出字母a-z,

要求A B 线程交替执行,最终输出如下图所示

在这里插入图片描述

指定具体执行顺序

要求A B 线程交替执行,并且必须是A线程先执行,B后执行

解决思路:初始指定一个标志位,标记哪个线程可以运行,只要当前标志位不是当前线程(不满足运行条件),就一直等待

伪代码

线程A{
	while(满足当前线程的运行条件)
		不满足,则直接释放锁并进入wait状态
	执行当前线程内容
	通知下一个执行的线程,让其运行条件为true
}

实现一:wait_notify机制 + Thread标志位

实现思路:

  • 用标志位Thread标识哪个先执行
  • 线程A:只要标志位不是A就不满足运行条件,就等待;否则才执行内容,并切换Thread为下一个要执行的线程B
  • 线程A:只要标志位不是B就不满足运行条件,就等待;否则才执行内容,并切换Thread为下一个要执行的线程A

拓展:如果是三个线程交替,只需要让B线程执行完后把threadName标记为C,然后让C线程执行完后把threadName标记为A即可

public class AlternateThreads {

    private static final Object lock = new Object();
    private static String threadName = "A";

    public static void main(String[] args) {
        Thread threadA = new Thread(() -> {
            synchronized (lock) {
                for (int i = 1; i <= 26; i++) {
                    //threadName只要不是线程A,就等待
                    while (threadName!="A") {
                        try {
                            lock.wait();
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                    System.out.print(i + " ");
                    threadName = "B";
                    lock.notify();
                }
            }
        });

        Thread threadB = new Thread(() -> {
            synchronized (lock) {
                for (char c = 'a'; c <= 'z'; c++) {
                    //threadName只要不是线程A,就等待
                    while (threadName!="B") {
                        try {
                            lock.wait();
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                    System.out.print(c + " ");
                    threadName = "A";
                    lock.notify();
                }
            }
        });

        threadA.start();
        threadB.start();
    }
}

实现二:lock_condition机制 + Thread标志位

实现思路:和wait_notiy实现的基本一致,也是,只不过wait变成了await,notify变成了signal

拓展:如果是三个线程交替,只需要让B线程执行完后把threadName标记为C,然后让C线程执行完后把threadName标记为A即可

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class AlternateThreads {

    private static Lock lock = new ReentrantLock();
    private static Condition condition = lock.newCondition();
    private static String threadName = "A";

    /**目标:交替执行的基础上,必须先执行A,再执行B
     * 方案:lock_condition机制 + Thread标志位
     * 实现思路:和我wait_notiy实现的基本一致,也是,只不过wait变成了await,notify变成了signal
     * @param args
     */
    public static void main(String[] args) {
        Thread threadA = new Thread(() -> {
            try {
                lock.lock();
                for (int i = 1; i <= 26; i++) {
                    while (threadName!="A") {
                        condition.await();
                    }
                    System.out.print(i + " ");
                    threadName = "B";
                    condition.signal();
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                lock.unlock();
            }
        });

        Thread threadB = new Thread(() -> {
            try {
                lock.lock();
                for (char c = 'a'; c <= 'z'; c++) {
                    while (threadName!="B") {
                        condition.await();
                    }
                    System.out.print(c + " ");
                    threadName = "A";
                    condition.signal();
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                lock.unlock();
            }
        });

        threadA.start();
        threadB.start();
    }
}

实现三:semaphore信号量

实现思路:

  • 线程A:开始执行时,先申请信号量s1,执行结束后,再释放信号量s2,让B结束等待;
  • 线程B:开始执行时,先申请信号量s2,执行结束后,再释放信号量s1,让A结束等待;
  • 初始化:设置线程A对应的信号量初始为1,B线程对应信号量为0,保证A先执行

拓展:如果是三个线程交替,只需要让B线程执行完后release信号量C,然后让C线程执行完后release信号量A

public class AlternateThreads {

    private static Semaphore semaphoreA = new Semaphore(1);
    private static Semaphore semaphoreB = new Semaphore(0);

    public static void main(String[] args) {
        Thread threadA = new Thread(() -> {
            for (int i = 1; i <= 26; i++) {
                try {
                    semaphoreA.acquire();//申请信号量A
                    System.out.print(i + " ");
                    semaphoreB.release();//通知B可以运行了
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });

        Thread threadB = new Thread(() -> {
            for (char c = 'a'; c <= 'z'; c++) {
                try {
                    semaphoreB.acquire();//申请信号量B
                    System.out.print(c + " ");
                    semaphoreA.release();//通知A可以运行了
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });

        threadA.start();
        threadB.start();
    }
}

不指定具体执行顺序,只交替执行

要求:如果不要求A和B哪个先执行,只要求A和B交替执行

解决思路:保证互斥,直接执行内容,再notify唤醒其它线程,最后自己再释放锁进入wait状态

伪代码

线程A{
	while(是否满足当前线程A的运行条件?)
		不满足,则直接释放锁并进入wait
	执行当前线程内容
	通知下一个执行的线程B,让其运行条件为true
}

为什么这样不能保证哪个先执行?

因为各个线程只要竞争到锁,就会直接运行,后续再wait,而不是先判断是否要wait

实现一:synchronized和wait、notify实现

public class ThreadSyn {
    //实现方式一,synchronized和wait、notify实现,不保证先执行哪一个
    public static void test1() {
        Object lock = new Object();
        boolean flag = false;
        Thread a = new Thread(() -> {
            synchronized (lock) {
                for (int i = 1; i <= 26; i++) {
                    //直接运行
                    System.out.print(i + " ");
                    try {
                        //先唤醒,再阻塞
                        lock.notify();
                        lock.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                lock.notify();//结束后唤醒,防止死锁
            }
        });
        Thread b = new Thread(() -> {
            synchronized (lock) {
                for (char ch = 'a'; ch <= 'z'; ch++) {
                    //直接运行
                    System.out.print(ch + " ");
                    try {
                        //先唤醒,再阻塞
                        lock.notify();
                        lock.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                lock.notify();//结束后唤醒,防止死锁
            }
        });
        a.start();
        b.start();
    }

实现二:lock和condition实现

    //实现方式二:lock和condition实现,A B交替执行,也不确定先执行的是哪个,只能确保轮流执行
    public static void test2() {
        Lock lock = new ReentrantLock();
        Condition condition_A = lock.newCondition();
        Condition condition_B = lock.newCondition();
        Thread a = new Thread(() -> {
            lock.lock();
            for (int i = 1; i <= 26; i++) {
                //直接运行
                System.out.print(i + " ");
                try {
                    //先唤醒,再阻塞
                    condition_B.signal();//唤醒线程B
                    condition_A.await();//等待条件A
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            condition_B.signal();//执行完for循环后需释放最后在等待的B线程,防止死锁
            lock.unlock();

        });
        Thread b = new Thread(() -> {
            lock.lock();
            for (char ch = 'a'; ch <= 'z'; ch++) {
                System.out.print(ch + " ");//执行线程操作
                try {
                    condition_A.signal();//唤醒线程A
                    condition_B.await();//等待条件B
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            condition_A.signal();//执行完for循环后需释放最后在等待的A线程,防止死锁
            lock.unlock();
        });

        a.start();
        b.start();
    }

总结

如果想要指定执行顺序,则需要用信号量,或者自己实现一个标志位去模拟信号量

如果不需要指定执行顺序,则谁先竞争到锁就谁先执行,执行完后唤醒另一个线程,再进入wait状态

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/623102.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

AlphaFold 3:开启生物医药新革命

AlphaFold 3简介 DeepMind与Isomorphic Labs联合发布了AlphaFold 3&#xff0c;这是一个可以更准确预测蛋白质和其他生物分子结构及其相互作用的AI模型&#xff0c;标志着生物医学研究的新革命 AlphaFold 3&#xff0c;这款由DeepMind与Isomorphic Lab联手推出的最新人工智能…

HIVE卡口流量需求分析

HIVE卡口流量需求分析 目录 HIVE卡口流量需求分析 1.创建表格 插入数据 2.需求 3.总结&#xff1a; 1.创建表格 插入数据 CREATE TABLE learn3.veh_pass( id STRING COMMENT "卡口编号", pass_time STRING COMMENT "进过时间", pass_num int COMMENT …

Python---Pandas万字总结(1)

Pandas基础-1 Pandas 是 一个强大的分析结构化数据的工具集。Pandas 以 NumPy 为基础&#xff08;实现数据存储和运算&#xff09;&#xff0c;提供了专门用于数据分析的类型、方法和函数&#xff0c;对数据分析和数据挖掘提供了很好的支持&#xff1b;同时 pandas 还可以跟数…

微软: 用于文本到语音合成(TTS)的语言模型方法VALL-E

微软引入了一种用于文本到语音合成(TTS)的语言模型方法。具体而言,微软使用从现成的神经音频编解码器模型中得到的离散编码训练了一个神经编解码器语言模型(称为VALL-E),并将TTS视为条件语言建模任务,而不是像之前的工作那样进行连续信号回归。在预训练阶段,微软将TTS训练数据扩…

JavaSE——集合框架一(2/7)-Collection集合的遍历方式-迭代器、增强for循环、Lambda、案例

目录 Collection的遍历方式 迭代器 增强for循环&#xff08;foreach&#xff09; Lambda表达式遍历集合 案例 需求与分析 代码部分 运行结果 Collection的遍历方式 迭代器 选代器是用来遍历集合的专用方式&#xff08;数组没有选代器&#xff09;&#xff0c;在Java中…

git-将本地项目上传到远程仓库

在gitee中新建一个远程仓库。 填写对应内容。 打开你想上传的文件夹&#xff0c;比如我想上传yuanshen 右击&#xff0c;打开git bash 输入git init初始化仓库 git init 添加项目所有文件 git add . 将添加的文件提交到本地仓库&#xff08;提交说明必填&#xff09; git …

怎么获得公网IP?

什么是公网IP 在计算机网络中&#xff0c;公网IP&#xff08;Internet Protocol&#xff09;是指可以直接被互联网访问和通信的IP地址。相对应的&#xff0c;私网IP则是在局域网内使用的IP地址&#xff0c;无法直接被互联网访问。获得公网IP对于一些特定的网络需求非常重要&am…

主机扫漏:Apache Tomcat 环境问题漏洞(CVE-2023-46589)

文章目录 引言I 修复此安全问题see also引言 Apache Tomcat存在环境问题漏洞,该漏洞源于存在不正确的输入验证漏洞,可能会导致将单个请求视为多个请求,从而在反向代理后面出现请求走私。 Tomcat did not correctly parse HTTP trailer headers. A specially crafted traile…

【LAMMPS学习】八、基础知识(6.5)PyLammps 教程

8. 基础知识 此部分描述了如何使用 LAMMPS 为用户和开发人员执行各种任务。术语表页面还列出了 MD 术语&#xff0c;以及相应 LAMMPS 手册页的链接。 LAMMPS 源代码分发的 examples 目录中包含的示例输入脚本以及示例脚本页面上突出显示的示例输入脚本还展示了如何设置和运行各…

Poetry Camera照相机将照片转换成诗歌并打印出来;吴恩达新课程深入了解Mistral;科学研究AI小助手data-to-paper

✨ 1: Poetry Camera 将拍摄的照片转换成诗歌并打印出来 Poetry Camera——一个能够把它所见之物转化成诗歌并打印出来的相机。你在一个美丽的公园&#xff0c;或者是一个充满故事的老街道。只要用Poetry Camera拍下这一刻&#xff0c;它就能立刻给你一首关于这个场景的诗。 …

(项目)-KDE巡检报告(模板

金山云于12月26日对建行共计【30】个KDE集群,合计【198】台服务器进行了巡检服务。共发现系统风险【135】条,服务风险【1912】条,服务配置风险【368】条。 一、系统风险 1、风险分析(图片+描述) (1)磁盘使用率高 问题描述多个集群的多台服务器磁盘使用率较高,远超过…

ModuleSim 仿真找不到模块 module is not defined

提示如下&#xff1a; # vsim -t 1ps -L altera_ver -L lpm_ver -L sgate_ver -L altera_mf_ver -L altera_lnsim_ver -L cycloneive_ver -L rtl_work -L work -voptargs""acc"" pulse_generator_tb # Start time: 14:26:25 on May 10,2024 # ** Note: (…

图的概念、性质和存储与简单遍历

前置知识&#xff1a;树的基本概念及性质 为了保证学习效果&#xff0c;请保证已经掌握前置知识之后&#xff0c;再来学习本章节&#xff01;如果在阅读中遇到困难&#xff0c;也可以回到前面章节查阅。 学习目标 掌握图的基本概念掌握图的一些性质 图的概念 基本概念 图 (…

绝地求生:经典艾伦格即将回归!绝地求生艾伦格进化史

29.2版本经典艾伦格地图将会回归&#xff0c;让我回顾一下艾伦格地图的改动历史吧&#xff01; 回归时间 2016年早期A测 A测 4.1版本&#xff1a;艾伦格-新视界 主要区域变动 Military Base(军事基地) Military Base Mylta Power&#xff08;大电&#xff09; Mylta Power …

Adobe After Effects AE v24.3.0 解锁版 (视频合成及视频特效制作)

Adobe系列软件安装目录 一、Adobe Photoshop PS 25.6.0 解锁版 (最流行的图像设计软件) 二、Adobe Media Encoder ME v24.3.0 解锁版 (视频和音频编码渲染工具) 三、Adobe Premiere Pro v24.3.0 解锁版 (领先的视频编辑软件) 四、Adobe After Effects AE v24.3.0 解锁版 (视…

笔记:完善python selenium 讯飞写作的整体自动化

昨天做得不太好,今天再来一次,我发现,只要写得多,一定会有发现。 1、加入本地目录,不要一直登录。 # 定义Edge浏览器的用户数据目录edge_user_data_dir = r"C:\Users\Administrator\AppData\Local\Microsoft\Edge\User Data\Default"# 设置Edge选项edge_optio…

解锁AI写作新纪元的文心一言指令

解锁AI写作新纪元的文心一言指令 在人工智能&#xff08;AI&#xff09;飞速发展的今天&#xff0c;自然语言处理&#xff08;NLP&#xff09;技术取得了显著的进步。文心一言&#xff0c;作为NLP领域的一颗璀璨明星&#xff0c;以其强大的文本生成和指令理解能力&#xff0c;为…

广东省网络安全竞赛部分web,misc wp

我的队伍只做了5题&#xff0c;还是太菜了&#xff0c;本来不想发的&#xff0c;但是写都写了&#xff0c;还是水一篇博客吧 这里是我们队的wp misc1 给了一个压缩包&#xff0c;解压需要密码&#xff0c;用纯数字密码没跑出来&#xff0c;感觉可能不是要强跑&#xff0c;看…

UIKit常用API:Transform

需求 使用Transform系列的API&#xff0c;该API中提供了旋转、平移等功能。其中函数中带make的效果是只变化一次&#xff0c;不带make可变化多次。此外&#xff0c;还有恢复函数&#xff1a;CGAffineTransformIdentity。 代码实现 注意按钮绑定的是同一个响应事件&#xff0…

详细分清Session,Cookie和Token之间的区别,以及JWT是什么东西

Cookie Cookie是一种小型的文本文件&#xff0c;由网站在用户访问时存储在其计算机或移动设备上&#xff0c;Cookie主要用于跟踪、识别和存储有关用户的信息。 简单来说Cookie就是用来存储某些后端发送给前端的数据&#xff0c;例如我们登陆后&#xff0c;后端会返回一个登录…