java多线程(10)线程通信

  1. Object类(并非Thread类)有三个方法:wait(), notify(),notifyAll()可以用于多线程通信
    1. wait是将当前线程等待,知道其他线程调用该同步监视器的notify方法或者notifyAll方法来唤醒该现线程
    2. notify是唤醒这个同步监视器上等待的单个线程,但是如果有多个线程再这个同步监视器上等待,那么这个方法唤醒的那个线程将是随机的
    3. notifyAll唤醒再次同步监视器上等待的所有线程
  2. 以下示例为一个模拟存钱取钱操作来实现多线程之间的通信
    1. 通过布尔值flag来决定是存钱操作等待还是取钱操作等待。举例:当flag为假,表示账号里没有钱,这个时候取钱操作将等待(wait()),然后到了存钱操作,因为flag为假,所以可以存钱,再进行完存钱操作之后,将flag置为真,并且唤醒等待的所有线程,这里一个存钱操作就进行完毕了。取钱操作就被唤醒,再次判断flag,flag为真,可以取钱,取钱之后将flag置为假,唤醒其他在等待的所有线程,这是有可能就唤醒了执行到一般正在等待的其他存钱线程
  3. 以下为代码示例,
    import java.util.Objects;
    
    public class Account1 {
        private String accountNo;
        private  double balance;
        private boolean flag = false;
        public Account1(){}
        public Account1(String accountNo, double balance){
            this.accountNo = accountNo;
            this.balance = balance;
        }
    
        public String getAccountNo() {
            return accountNo;
        }
    
        public void setAccountNo(String accountNo) {
            this.accountNo = accountNo;
        }
    
        public double getBalance(){
            return this.balance;
        }
    
        public synchronized void draw(double drawAccount){
            try{
                
    //            如果flag为假,表示账号里没有钱,取钱方法阻塞
                if(!flag){
                    wait();
                }else {
                    System.out.println(Thread.currentThread().getName() + " 取钱" + drawAccount) ;
                    balance -= drawAccount;
                    System.out.println("账号余额为" + " " + balance);
                    
                    
    //                如果flag为真,表示账号里有钱,进行取钱操作,并且将flag置为假之后唤醒其他线程
                    flag = false;
                    notifyAll();
                    
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        }
    
        public synchronized void deposit(double depositAccount){
            try{
    
    //            如果flag为真,表示账号里有钱,存钱方法阻塞
                if(flag){
                    wait();
                }else {
                    System.out.println(Thread.currentThread().getName() + " 存款:" + depositAccount);
                    balance += depositAccount;
                    System.out.println("账户余额为: " + balance);
    
    //                如果flag为假,表示账号里没钱,进行存钱操作,并且将flag置为真之后唤醒其他线程 
                    flag = true;
                    notifyAll();
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        }
    
        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            Account1 account1 = (Account1) o;
            return Double.compare(account1.balance, balance) == 0 &&
                    flag == account1.flag &&
                    Objects.equals(accountNo, account1.accountNo);
        }
    
        @Override
        public int hashCode() {
            return Objects.hash(accountNo, balance, flag);
        }
    }
    
    public class DrawThread1  extends  Thread{
        private Account1 account1;
        private double drawAmount1;
        public DrawThread1(String name, Account1 account1, double drawAmount1){
            super(name);
            this.account1 = account1;
            this.drawAmount1 =  drawAmount1;
        }
    
        public void run(){
            for(int  i=0 ; i<100; i++){
                account1.draw(drawAmount1);
            }
        }
    
    }
    
    public class DespositThread extends  Thread{
        private Account1 account1;
        private double depositAmount;
        public DespositThread(String name, Account1 account1, double depositAmount){
            super(name);
            this.account1 = account1;
            this.depositAmount = depositAmount;
        }
    
        public void run(){
            for(int i=0; i<100; i++){
                account1.deposit(depositAmount);
            }
        }
    }
    
    public class DrawTest1 {
        public static void main(String args[]){
            Account1 account1 = new Account1("1234567", 0);
            new DrawThread1("取钱者", account1, 800).start();
            new DespositThread("存款者甲", account1,800).start();
            new DespositThread("存款者乙", account1,800).start();
            new DespositThread("存款者丙", account1,800).start();
        }
    }
    

     

  4. 上述的是代码代码块的线程通信,实际上还有一张是通过同步锁的实现的线程通信,java提供了一个Condition类来实现线程通信的。Condition示例被绑定在一个同步锁Lock的对象上,它也提供了类似上面说的三种实现线程通信的方法

    1. await类似于上面的wait方法。导致当前线程等待

    2. signal唤醒在此同步锁对象上等待的一个线程,如果又多个线程在这个同步锁上等待,那么它会随机的唤醒一个线程

    3. signalAll唤醒在此同步锁上等待的所有线程

  5. 以下为代码示例,具体的讲解穿插在代码中

    import java.util.Objects;
    import java.util.concurrent.locks.Condition;
    import java.util.concurrent.locks.Lock;
    import java.util.concurrent.locks.ReentrantLock;
    
    public class Account1 {
        private String accountNo;
        private  double balance;
        private boolean flag = false;
    
    //    这两步为实例化一个同步锁对象,并且将一个Condition对象绑定在同步锁对象上
        private final Lock lock = new ReentrantLock();
        private final Condition condition =  lock.newCondition();
        
    
        public Account1(){}
        public Account1(String accountNo, double balance){
            this.accountNo = accountNo;
            this.balance = balance;
        }
    
        public String getAccountNo() {
            return accountNo;
        }
    
        public void setAccountNo(String accountNo) {
            this.accountNo = accountNo;
        }
    
        public double getBalance(){
            return this.balance;
        }
    
        public  void draw(double drawAccount){
            
            lock.lock();
    //        同步锁将当前以下内容加锁
    
            try{
    
    //            如果flag为假,表示账号里没有钱,取钱方法阻塞
                if(!flag){
                    
                    condition.await();
    //                这里就相当于上面程序的wait(),以下有关condition对象的操作都和上面程序的操作是
    //                一一对应的
                    
                }else {
                    System.out.println(Thread.currentThread().getName() + " 取钱" + drawAccount) ;
                    balance -= drawAccount;
                    System.out.println("账号余额为" + " " + balance);
    
    
    //                如果flag为真,表示账号里有钱,进行取钱操作,并且将flag置为假之后唤醒其他线程
                    flag = false;
                    condition. signalAll();
    
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }finally{
                lock.unlock();
            }
        }
    
        public  void deposit(double depositAccount){
            lock.lock();
            try{
    
    //            如果flag为真,表示账号里有钱,存钱方法阻塞
                if(flag){
                    condition.await();
                }else {
                    System.out.println(Thread.currentThread().getName() + " 存款:" + depositAccount);
                    balance += depositAccount;
                    System.out.println("账户余额为: " + balance);
    
    //                如果flag为假,表示账号里没钱,进行存钱操作,并且将flag置为真之后唤醒其他线程
                    flag = true;
                    condition.signalAll();
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }finally {
                lock.unlock();
            }
        }
    
        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            Account1 account1 = (Account1) o;
            return Double.compare(account1.balance, balance) == 0 &&
                    flag == account1.flag &&
                    Objects.equals(accountNo, account1.accountNo);
        }
    
        @Override
        public int hashCode() {
            return Objects.hash(accountNo, balance, flag);
        }
    }
    

    这是我看李刚编著的《疯狂Java讲义》后总结出来的。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值