Java多线程 - 线程安全和线程同步解决线程安全问题

这篇具有很好参考价值的文章主要介绍了Java多线程 - 线程安全和线程同步解决线程安全问题。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

线程安全问题

线程安全问题指的是: 多个线程同时操作同一个共享资源的时候可能会出现业务安全问题,称为线程安全问题。

举例: 取钱模型演示

需求:小明和小红是一对夫妻,他们有一个共同的账户,余额是10万元。

如果小明和小红同时来取钱,而且2人都要取钱10万元,可能出现什么问题呢?

Java多线程 - 线程安全和线程同步解决线程安全问题

在取钱之前都需要判断余额是否足够, 例如两个线程同时执行, 两个线程都进行了余额判断, 发现余额充足;

此时小明线程取走10万, 账户余额为0; 小红线程由于此时已经判断过余额, 继续取钱的时候就不会继续判断余额, 直接将余额取出来; 那么两个人都取走了10万, 银行就亏了10万, 这就是多线程带来的安全问题

线程安全问题模拟, 我们将上面的例子用代码模拟出来多线程的安全隐患:

需求:

小明和小红是一对夫妻,他们有一个共同的账户,余额是10万元,模拟2人同时去取钱10万。

分析:

  1. 需要提供一个账户类,创建一个账户对象代表2个人的共享账户。
  2. 需要定义一个线程类,线程类可以处理账户对象。
  3. 创建2个线程对象,传入同一个账户对象。
  4. 启动2个线程,去同一个账户对象中取钱10万。

实现步骤:

模拟一个账户类给小明线程和小红线程, 我们模拟关键信息即可

public class Account {
    private double money;

    public  Account() {}

    public Account(double money) {
        this.money = money;
    }

    /**
        取钱方法
     */
    public void drawMoney(double money) {
        // 获取取钱人的名字
        String name = Thread.currentThread().getName();
        if (this.money >= money) { // 判断余额是否充足
            System.out.println(name + "取走了" + money + "元");
            this.money -= money;
        } else {
            System.out.println("余额不足");
        }
    }

    public double getMoney() {
        return money;
    }

    public void setMoney(double money) {
        this.money = money;
    }
}

定义一个线程类用来处理账户对象

public class DrawThread extends Thread {
    private Account acc;

    public DrawThread(Account acc, String name) {
        super(name);
        this.acc = acc;
    }

    @Override
    public void run() {
        acc.drawMoney(100000);
    }
}

在主类中, 创建2个线程对象,传入同一个账户对象; 启动2个线程,去同一个账户对象中取钱10万

public class ThreadDemo {
    public static void main(String[] args) throws Exception {
        // 创建账户对象
        Account acc = new Account(100000);

        // 创建两个子线程, 并启动线程
        new DrawThread(acc, "小明").start(); // 小明取走了100000.0元
        new DrawThread(acc, "小红").start(); // 小红取走了100000.0元

        // 主线程睡眠两秒后, 查看余额
        Thread.sleep(2000);
        System.out.println(acc.getMoney()); // -100000.0
    }
}

线程同步

为了解决上面线程安全的问题。

取钱案例出现问题的原因

多个线程同时执行,发现账户都是够钱的。

如何才能保证线程安全呢

让多个线程实现先后依次排队访问共享资源,这样就解决了安全问题

线程同步的核心:

线程同步的核心是加锁,把共享资源进行上锁,每次只能一个线程进入访问完毕以后解锁,然后其他线程才能进来。

加锁的方式有三种方式: 同步代码块, 同步方法和Lock锁, 下面我们来分别学习加锁的方法

方式一: 同步代码块

作用:将出现线程安全问题的核心代码给上锁

原理:每次只能一个线程进入,执行完毕后自动解锁,其他线程才可以进来执行。

同步代码块上锁的格式:

synchronized(同步锁对象) {
  	// 操作共享资源的代码(核心代码)
}

例如我们给上面模拟的线程安全例子中的核心代码进行加锁操作:

进行加锁操作就解决了线程安全问题

public class Account {
    private double money;

    public  Account() {}

    public Account(double money) {
        this.money = money;
    }

    /**
        取钱方法
     */
    public void drawMoney(double money) {
        String name = Thread.currentThread().getName();

        // 对核心代码进行加锁, 这里锁对象随便使用了一个字符串模拟(无实际代表意义)
        synchronized ("chen") {
            if (this.money >= money) {
                System.out.println(name + "取走了" + money + "元");
                this.money -= money;
            } else {
                System.out.println("余额不足");
            }
        }
    }

    public double getMoney() {
        return money;
    }

    public void setMoney(double money) {
        this.money = money;
    }
}

锁对象注意:

理论上:锁对象只要对于当前同时执行的线程来说是唯一的一个对象即可。

但是实际上使用任意唯一的锁对象并不好, 会影响其他无关线程的执行, 例如上面例子中, 会将其他无关的账户也锁起来。

锁对象的规范要求:

规范上:建议使用共享资源作为锁对象。

  • 对于实例方法中, 建议使用this作为锁对象。
// 加锁, 使用共享资源作为锁对象
synchronized (this) {
}
  • 对于静态方法中, 建议使用字节码(类名.class)对象作为锁对象。
// 加锁, 使用共享资源作为锁对象
synchronized (类名.class) {
}

方式二: 同步方法

作用: 将出现线程安全问题的核心方法给上锁

原理:每次只能一个线程进入,执行完毕以后自动解锁,其他线程才可以进来执行。

同步方法的上锁格式:

修饰符 synchronized 返回值类型 方法名称(形参列表) {
	  // 操作共享资源的代码
}

同步方法上锁演示代码:

// 给核心方法上锁
public synchronized void drawMoney(double money) {
    String name = Thread.currentThread().getName();

    if (this.money >= money) {
        System.out.println(name + "取走了" + money + "元");
        this.money -= money;
    } else {
        System.out.println("余额不足");
    }
}

同步方法底层原理:

同步方法其实底层也是有隐式锁对象的,只是锁的范围是整个方法代码。

如果方法是实例方法:同步方法默认用this作为的锁对象。但是代码要高度面向对象!

如果方法是静态方法:同步方法默认用类名.class作为的锁对象。

方式三: Lock锁

Lock锁

为了更清晰的表达如何加锁和释放锁,JDK5以后提供了一个新的锁对象Lock,更加灵活、方便。

Lock实现提供比使用synchronized方法和语句可以获得更广泛灵活的锁定操作。

Lock是接口不能直接实例化,这里采用它的实现类ReentrantLock来构建Lock锁对象。

方法名称 说明
ReentrantLock() 获得Lock锁的实现类对象

Lock的API:

方法名称 说明
lock() 获得锁
unlock() 释放锁

Lock使用演示代码:

基本使用如下, 我们可以看出Lock使用是非常灵活的

public class Account {
    private double money;
    // 定义一个示例变量锁对象, 每创建一个类就会创建一个锁对象, 加final修饰, 表示不可替换
    private final Lock lock = new ReentrantLock();

    public  Account() {}
    public Account(double money) {
        this.money = money;
    }
    
    public synchronized void drawMoney(double money) {
        String name = Thread.currentThread().getName();

        // 调用锁对象上锁
        lock.lock();
        if (this.money >= money) {
            System.out.println(name + "取走了" + money + "元");
            this.money -= money;
        } else {
            System.out.println("余额不足");
        }
        // 调用锁对象解锁
        lock.unlock();
    }

    public double getMoney() {
        return money;
    }

    public void setMoney(double money) {
        this.money = money;
    }
}

但是如果我们上锁和解锁之间的代码出现了异常, 永远都不会执行解锁操作, 所以更严谨的写法是将解锁的操作放到try…finally中, 保证会执行解锁的操作文章来源地址https://www.toymoban.com/news/detail-414167.html

public synchronized void drawMoney(double money) {
    String name = Thread.currentThread().getName();

    // 调用锁对象上锁
    try {
        lock.lock();
        if (this.money >= money) {
            System.out.println(name + "取走了" + money + "元");
            this.money -= money;
        } else {
            System.out.println("余额不足");
        }
    } finally {
        // 调用锁对象解锁
        lock.unlock();
    }
}

到了这里,关于Java多线程 - 线程安全和线程同步解决线程安全问题的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Java中的多线程——线程安全问题

    作者:~小明学编程   文章专栏:JavaEE 格言:热爱编程的,终将被编程所厚爱。 目录 多线程所带来的不安全问题 什么是线程安全 线程不安全的原因 修改共享数据 修改操作不是原子的 内存可见性对线程的影响 指令重排序 解决线程不安全的问题 synchronized 互斥 刷新内

    2024年02月03日
    浏览(55)
  • 【Java】线程安全问题

    在之前的文章中,已经介绍了关于线程的基础知识。 我的主页: 🍆🍆🍆爱吃南瓜的北瓜 欢迎各位大佬来到我的主页进行指点 一同进步!!! 我们创建两个线程t1和t2,对静态变量count执行++操作各50000次。 我们的预期结果是100000。但是当两个线程分别执行++操作时最后的结果

    2024年04月10日
    浏览(28)
  • 【多线程】线程安全问题原因与解决方案

    目录 线程安全的概念 线程不安全示例 线程不安全的原因      多个线程修改了同一个变量     线程是抢占式执行的     原子性     内存可见性     有序性 线程不安全解决办法  synchronized -监视器锁monitor lock     synchronized 的特性         互斥         刷新内

    2024年02月06日
    浏览(24)
  • 线程安全问题及解决方法

    线程在执行的过程中出现错误的主要原因有以下几种: 1、根本原因 导致线程不安全的所有原因中,最根本的原因是——抢占式执行。因为CPU字在进行线程调度的时候,是随机调度的,而且这是无法避免的一种原因。 2、代码结构 当多个线程同时修改同一个变量的时候,很容

    2024年02月06日
    浏览(21)
  • Java多线程【状态与安全问题】

    线程状态 说明 NEW 安排了工作, 还未开始行动 RUNNABLE 可工作的. 又可以分成正在工作中和即将开始工作 BLOCKED 这几个都表示排队等着其他事情 WAITING 这几个都表示排队等着其他事情 TIMED_WAITING 这几个都表示排队等着其他事情 TERMINATED 工作完成了 1.NEW状态:安排了工作, 还未开始

    2023年04月09日
    浏览(33)
  • 【线程安全】死锁问题及解决方案

    比如上一次讲到 synchronized 的时候,一个线程,对同一个对象连续加锁两次,如果出现阻塞等待,代表这个锁是不可重入锁,这样的线程,也就称为死锁! 一旦程序进入死锁了就会导致线程僵住了,无法继续执行后续的工作了,程序也就出现了严重的 BUG! 而死锁这样的情况

    2024年02月06日
    浏览(34)
  • Java多线程基础-8:单例模式及其线程安全问题

    单例模式是经典的设计模式之一。什么是设计模式?代码的设计模式类似于棋谱,棋谱就是一些下棋的固定套路,是前人总结出来的一些固定的打法。依照棋谱来下棋,不说能下得非常好,但至少是有迹可循,不会下得很糟糕。代码的设计模式也是一样。 设计模式,就是软件

    2024年02月05日
    浏览(35)
  • 线程安全问题的原因和解决方案

    如果某个代码,在单线程执行下没有问题,在多线程执行下执行也没有问题,则称“线程安全”,反之称“线程不安全”。 目录 前言 一、简述线程不安全案例 二、线程安全问题的原因 (一)(根本问题)线程调度是随机的 (二)代码的结构问题 (三)代码执行不是原子的

    2024年02月14日
    浏览(28)
  • 【Java|多线程与高并发】线程安全问题以及synchronized使用实例

    Java多线程环境下,多个线程同时访问共享资源时可能出现的数据竞争和不一致的情况。 线程安全一直都是一个令人头疼的问题.为了解决这个问题,Java为我们提供了很多方式. synchronized、ReentrantLock类等。 使用线程安全的数据结构,例如ConcurrentHashMap、ConcurrentLinkedQueue等

    2024年02月09日
    浏览(30)
  • Java中SimpleDateFormat的线程安全性问题

    在日常开发中,我们经常会用到时间,我们有很多办法在Java代码中获取时间。但不同的方法获取到的时间格式不尽相同,这时就需要一种格式化工具,把时间显示成我们需要的格式,最常用的方法就是使用SImpleDateFormat类。这是一个看上去功能比较简单的类,但使用不当,也

    2024年01月25日
    浏览(31)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包