java多线程 synchronized

synchronized 设置为同步方法

package synchronizedTest;

//模拟买票
public class UnsafeBuyTicket {
    public static void main(String[] args) {
        BuyTicket buyTicket = new BuyTicket();

        new Thread(buyTicket,"用户A").start();
        new Thread(buyTicket,"用户B").start();
        new Thread(buyTicket,"用户C").start();
    }
}
//不安全
class BuyTicket implements Runnable{
//  设置票数
    private static int ticketNums = 10;

    @Override
    public void run() {
        while (true){
            if(ticketNums<0){
                break;
            }
            try {
                System.out.println(Thread.currentThread().getName()+"买到了票,剩余" + --ticketNums + "张票");
                Thread.sleep(100);//防止票被一个人买完,模拟延迟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

}

在这里插入图片描述
发现线程并不安全
这时需要用到synchronized 设置为同步方法

class BuyTicket implements  Runnable{
    //  设置票数
    private static int ticketNums = 10;

    @Override
    public synchronized void run() {//只需要在这里加synchronized 据可以解决问题
        while (true){

            if(ticketNums<=0){
                break;
            }
            try {
                System.out.println(Thread.currentThread().getName()+"买到了票,剩余" + --ticketNums + "张票");
                Thread.sleep(100);//防止票被一个人买完,模拟延迟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

}

在这里插入图片描述

synchronized 设置为同步对象

package synchronizedTest;

//模拟银行取钱
public class Bank {
    public static void main(String[] args) {
        //账户
        Account account = new Account(100, "银行卡");

        Drawing a = new Drawing(account, 50, "A");
        Drawing b = new Drawing(account, 100, "B");

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

}

//账户
class Account{
    int money;
    String name;

    public Account(int money, String name) {
        this.money = money;
        this.name = name;
    }
}

//模拟银行取款
class Drawing extends Thread{
    //账户
    Account account;
    //取款金额
    int drawingMoney;

    public Drawing(Account account,int drawingMoney,String name){
        super(name);
        this.account = account;
        this.drawingMoney = drawingMoney;
    }

    @Override
    public void run() {
        //判断账户余额是否够取
        if(account.money - drawingMoney < 0){
            System.out.println(Thread.currentThread().getName()+"取钱失败,账户钱不够");
        	return;
        }

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        account.money -= drawingMoney;
//        this.getName() == Thread.currentThread().getName()
        System.out.println(this.getName()+"取了"+drawingMoney);
        System.out.println("账户余额:" + account.money);
    }
}

在这里插入图片描述

synchronized 锁的是this即Drawing,
当A进程进来时发现A对象Drawing没锁,则加锁运行run方法
当B进程进来时发现B对象Drawing没锁,则加锁运行run方法

(注意A和B的对象不同,所以不能锁run方法,应该锁共享资源即Account)

package synchronizedTest;

//模拟银行取钱
public class Bank {
    public static void main(String[] args) {
        //账户
        Account account = new Account(100, "银行卡");

        Drawing a = new Drawing(account, 50, "A");
        Drawing b = new Drawing(account, 100, "B");

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

}

//账户
class Account{
    int money;
    String name;

    public Account(int money, String name) {
        this.money = money;
        this.name = name;
    }
}

//模拟银行取款
class Drawing extends Thread{
    //账户
    Account account;
    //取款金额
    int drawingMoney;

    public Drawing(Account account,int drawingMoney,String name){
        super(name);
        this.account = account;
        this.drawingMoney = drawingMoney;
    }

    @Override
    public void run() {
        synchronized (account){
            //判断账户余额是否够取
            if(account.money - drawingMoney < 0){
                System.out.println(Thread.currentThread().getName()+"取钱失败,账户钱不够");
                return;
            }

            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            account.money -= drawingMoney;
//        this.getName() == Thread.currentThread().getName()
            System.out.println(this.getName()+"取了"+drawingMoney);
            System.out.println("账户余额:" + account.money);
        }
    }

}

在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值