Java教程

线程同步机制

本文主要是介绍线程同步机制,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

线程同步机制

  并发:同一个对象被多个线程同时操作

  

 

 

   线程同步:

    

  

 

   线程不安全案例1:

//不安全的买票    可能出现负数票,或者抢到了同一张票
public class UnsafeBuyTicket implements Runnable{

    private int ticketNums=10; //票
    boolean flag=true; //外部停止方式
    @Override
    public void run() {

        while (flag){
            buy();
        }

    }

    private void buy(){
        if(ticketNums<0){
            flag=false;
            return;
        }
        //模拟延时
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName()+"拿到了第"+ticketNums--+"票");
    }

    public static void main(String[] args) {

        UnsafeBuyTicket unsafeBuyTicket=new UnsafeBuyTicket();

        new Thread(unsafeBuyTicket,"学生").start();
        new Thread(unsafeBuyTicket,"老师").start();
        new Thread(unsafeBuyTicket,"黄牛党").start();
    }
}
 

  线程不安全案例2:

//不安全的取钱
//两个人去银行取钱
public class UnsafeBank {

    public static void main(String[] args) {

        Account account=new Account(100,"结婚基金");

        Drawing you=new Drawing(account,50,"你");
        Drawing girlFriend=new Drawing(account,100,"grilFriend");

        you.start();
        girlFriend.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; //取了多少钱
    int nowMoney;   //现在手里有多少钱

    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()+"钱不够了,取不了了");
        }
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //卡内余额=余额-你取的钱
        account.money=account.money-drawingMoney;
        //你手里的钱
        nowMoney=nowMoney+drawingMoney;

        System.out.println(account.name+"余额为:"+account.money);
        System.out.println(this.getName()+"手里的钱:"+nowMoney);
    }
}

  线程不安全案例3

import java.util.ArrayList;
import java.util.List;

public class UnsafeList {

    public static void main(String[] args) {

        List<String> list=new ArrayList<>();
        for(int i=0;i<10000;i++){

            new Thread(()->{
               list.add(Thread.currentThread().getName());
            }).start();
        }
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(list.size());
    }
}

同步方法

 

 

同步方法弊端

  

 

 

    

   解决方案一:在方法中使用  synchronized锁住方法  对应上面第一个案例

//不安全的买票
public class UnsafeBuyTicket implements Runnable{

    private int ticketNums=10; //票
    boolean flag=true; //外部停止方式
    @Override
    public void run() {

        while (flag){
            buy();
            //模拟延时
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

    }

    private synchronized void buy(){
        if(ticketNums<1){
            flag=false;
            return;
        }

        System.out.println(Thread.currentThread().getName()+"拿到了第"+ticketNums--+"票");
    }

    public static void main(String[] args) {

        UnsafeBuyTicket unsafeBuyTicket=new UnsafeBuyTicket();

        new Thread(unsafeBuyTicket,"学生").start();
        new Thread(unsafeBuyTicket,"老师").start();
        new Thread(unsafeBuyTicket,"黄牛党").start();
    }
}

  同步块  锁的对象就是变化的量,需要增删改的对象

  

//不安全的取钱
//两个人去银行取钱
public class UnsafeBank {

    public static void main(String[] args) {

        Account account=new Account(100,"结婚基金");

        Drawing you=new Drawing(account,50,"你");
        Drawing girlFriend=new Drawing(account,100,"grilFriend");

        you.start();
        girlFriend.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; //取了多少钱
    int nowMoney;   //现在手里有多少钱

    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=account.money-drawingMoney;
            //你手里的钱
            nowMoney=nowMoney+drawingMoney;

            System.out.println(account.name+"余额为:"+account.money);
            System.out.println(this.getName()+"手里的钱:"+nowMoney);
        }
    }

}

 

这篇关于线程同步机制的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!