美文网首页程序员
java同步线程(一)

java同步线程(一)

作者: RoundYuan | 来源:发表于2019-01-14 21:56 被阅读0次

    java同步线程(一)

               同步代码块
    

    线程安全问题,有一个经典的问题:

    银行取钱问题:
    1、账户类

    public class Account {
        //封装账户编号,账户余额两个Eield
        private String accountNo;
        private double balance;
        public Account(String accountNo,double balance)
        {
            this.accountNo=accountNo;
            this.balance=balance;
        }
        //创建的getter和setter方法
    
        public String getAccountNo() {
            return accountNo;
        }
    
        public void setAccountNo(String accountNo) {
            this.accountNo = accountNo;
        }
    
        public double getBalance() {
            return balance;
        }
    
        public void setBalance(double balance) {
            this.balance = balance;
        }
    
        @Override
        public int hashCode() {
            return accountNo.hashCode();
        }
    
        @Override
        public boolean equals(Object obj) {
            if (this==obj)
                return true;
            if (obj !=null&&obj.getClass()==Account.class)
            {
                Account target=(Account)obj;
                return target.getAccountNo().equals(accountNo);
            }
            return false;
        }
    
    }
    

    2、取钱的线程类:

    public class DrawThread extends Thread{
        private Account account;
        private double drawAmount;
    
        public DrawThread(String name,Account account,double drawAmount) {
            super(name);
            this.account=account;
            this.drawAmount=drawAmount;
        }
        //当多个线程修改同一个共享数据时,将涉及数据安全性问题
    
        @Override
        public void run() {
            
                if (account.getBalance()>=drawAmount)
                {
                    //突出钞票
                    System.out.println(getName()+"取钱成功!吐出钞票:"+drawAmount);
                    try {
                        Thread.sleep(1);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    account.setBalance(account.getBalance()-drawAmount);
                    System.out.println("\t余额为:"+account.getBalance());
    
                }
                else {
                    System.out.println(getName()+"取钱失败!余额不足!");
                }
            
    
        }
    }
    
    

    3、Test

    public class DrawTest {
        public static void main(String[] args) {
            //创建一个账户
            Account acct=new Account("123456",1000);
            //simulation two thread draw money from one account
            new DrawThread("甲",acct,800).start();
            new DrawThread("乙",acct,800).start();
        }
    }
    

    运行结果:

    同步1.png

    之所以会出现如图所示的结果,就是因为run()方法的方法体不具有同步安全性-程序中有两个并发线程
    在修改Account对象;在第一次输出后做切换后做线程切换,切换给另一个修改Account对象的线程,所以就出现了问题。

    为了解决这个问题,java的多线程支持引入了同步监视器解决这个问题,使用同步监视器的通用方法就是同步代码块。

    在执行同步代码块之前,必须取得对同步监视器的锁定。

    虽然java程序允许使用任何对象作为同步监视器,但想一下同步监视器的目的:阻止两个线程对同一个共享
    资源进行并发访问,因此通常推荐使用可能被并发访问的共享资源充当同步监视器。对于上面的模拟取钱程序,我们应该考虑
    使用账户(account)作为同步监视器。我们把程序修改成如下形式。

    public class DrawThread extends Thread{
        private Account account;
        private double drawAmount;
    
        public DrawThread(String name,Account account,double drawAmount) {
            super(name);
            this.account=account;
            this.drawAmount=drawAmount;
        }
        //当多个线程修改同一个共享数据时,将涉及数据安全性问题
    
        @Override
        public void run() {
            synchronized (account){
                if (account.getBalance()>=drawAmount)
                {
                    //突出钞票
                    System.out.println(getName()+"取钱成功!吐出钞票:"+drawAmount);
                    try {
                        Thread.sleep(1);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    account.setBalance(account.getBalance()-drawAmount);
                    System.out.println("\t余额为:"+account.getBalance());
    
                }
                else {
                    System.out.println(getName()+"取钱失败!余额不足!");
                }
            }
    
        }
    }
    
    

    这样就不会出错了

    相关文章

      网友评论

        本文标题:java同步线程(一)

        本文链接:https://www.haomeiwen.com/subject/vhdsdqtx.html