当前位置:主页 > 查看内容

Java-----多线程【并发与三大不安全案例】

发布时间:2021-05-19 00:00| 位朋友查看

简介:线程同步重点 并发 同一个对象 被 多个线程 同时操作 现实生活中我们会遇到”同一个资源多个人都想使用“的问题比如食堂排队打饭每个人都想吃饭最天然的解决办法就是排队一个个来。 处理多线程问题时多个线程访问同一个对象并且某些线程还想修改这个对象。这……

线程同步(重点)

并发

  • 同一个对象多个线程同时操作
  • 现实生活中,我们会遇到”同一个资源,多个人都想使用“的问题,比如,食堂排队打饭,每个人都想吃饭,最天然的解决办法就是,排队,一个个来。
  • 处理多线程问题时,多个线程访问同一个对象,并且某些线程还想修改这个对象。这时候我们就需要线程同步,线程同步其实就是一种等待机制,多个需要同时访问此对象的线程进入这个对象的等待池形成队列,等待前面线程使用完毕,下一个线程再使用。
  • 由于同一进程的多个线程共享同一块存储空间,在带来方便的同时,也带来了访问冲突的问题,为了保证数据在方法中被访问时的正确性,在访问时加入锁机制sychronized,当一个线程获得对象的排它锁,独占资源,其他线程必须等待,使用后释放锁即可.存在以下问题:
    • 一个线程持有锁会导致其他所有需要此锁的线程挂起;
    • 在多线程竞争下,加锁,释放锁会导致比较多的上下文切换 和 调度延时,引起性能问题;
    • 如果一个优先级高的线程等待一个优先级低的线程释放锁 会导致优先级倒置,引起性能问题.
    • 但是鱼和熊掌不能兼得

三大不安全案例

案例一:

public class UnsafeBuyTicket {

    public static void main(String[] args) {
        BuyTicket b = new BuyTicket();

        new Thread(b,"老黄牛").start();
        new Thread(b,"网友").start();
        new Thread(b,"自己").start();
    }
}

class BuyTicket implements Runnable{
    // 票
    private int ticetNum = 10;

    // 外部停止方式
    boolean flag = true;

    @Override
    public void run() {
        // 买票
        while (flag){
            buy();
        }
    }

    private void buy(){
        // 判断是否有票
        if(ticetNum<=0){
            flag = false;
            return;
        }

        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        // 买票
        System.out.println(Thread.currentThread().getName()+"拿到"+ticetNum--);
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Kw0YCqhK-1617838151276)(D:\学习\tupian\多线程\线程同步\cuo1.png)]

案例二:

// 不安全的取钱
// 两个人去银行账户,取钱
public class UnsafeBank {
    public static void main(String[] args) {

        // 账户
        Account account = new Account("项目基金",100);

        Drawing you = new Drawing(account,50,"组长");
        Drawing my = new Drawing(account,80,"组员");

        you.start();
        my.start();
    }
}

class Account{
     String name;
     int money;

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

// 银行:模拟取钱
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(this.getName()+"没钱了,取不了");
            return;
        }

        // sleep可以放大问题的发生性
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        // 卡内余额 = 余额 - 你取的钱
        account.money -= drawingMoney;

        // 你手里的钱
        nowMoney += drawingMoney;

        System.out.println(account.name+"余额为:"+account.money);

        System.out.println(this.getName()+"手里的钱"+nowMoney);

    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-v9CI9zk0-1617838151283)(D:\学习\tupian\多线程\线程同步\cuo2.png)]

案例三:

public class UnsafeList {
    public static void main(String[] args) {
        List<String> list = new ArrayList<String>();
        for(int i = 0;i<10000;i++){
            new Thread(()->{
                list.add(Thread.currentThread().getName());
            }).start();
        }

        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.out.println(list.size());
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BxwfbjqD-1617838151287)(D:\学习\tupian\多线程\线程同步\cuo3.png)]为什么会出现这样的情况呢?

因为集合添加的时候,如果中间有重复,它就会添加替换到同一个地方,导致循环走完数组数据不足的情况。

;原文链接:https://blog.csdn.net/AMT_MUZI/article/details/115500894
本站部分内容转载于网络,版权归原作者所有,转载之目的在于传播更多优秀技术内容,如有侵权请联系QQ/微信:153890879删除,谢谢!

推荐图文


随机推荐