线程安全(JAVA)

r囧r小猫 2024-02-19 09:38 100阅读 0赞

线程安全对于我们编写多线程代码是非常重要的。

什么是线程安全?

在我们平时的代码中有些代码在单线程程序中可以正常执行,但如果同样的代码放在在多个线程中执行就会引发BUG,而这种现象我们一般称为 “线程安全问题” 或 “线程不安全”。
例如:使用两个线程对 count 变量进行自增操作,每个线程10000次。

  1. private static int count;
  2. public static void main(String[] args) throws InterruptedException {
  3. Thread t1 = new Thread(()->{
  4. for (int i = 0; i < 10000; i++) {
  5. count++;
  6. }
  7. });
  8. Thread t2 = new Thread(()->{
  9. for (int i = 0; i < 10000; i++) {
  10. count++;
  11. }
  12. });
  13. t1.start();
  14. t2.start();
  15. t1.join();
  16. t2.join();
  17. System.out.println(count);
  18. }

在这里插入图片描述
结果可以看到和我们预期的并不相同,而且当我们多运行几次后,每次的结果还都不相同,这就是一个典型的 线程安全问题
为什么会出现上述情况呢?

  • 自增操作本质上其实分为三步
    – 从内存把数据读到 CPU
    – 进行加一操作
    – 把新数据写回到 CPU
  • 两个线程是并发执行

所以就会引发下面这种状况(程序按照时间线从上往下执行):
在这里插入图片描述
这里只是简单画了六种,由于线程的调度是无序的所以这里会有无数种情况,但是在这无数种情况中,只有当两个线程的调度每次都满足前两种情况才不会发生BUG。

引发线程安全的原因

一般引发线程安全都有以下原因:

  1. 操作系统中线程的调度是随机的(抢占式执行,罪魁祸首)
  2. 多个线程针对同一个变量进行修改
  3. 修改操作不是原子的
  4. 内存可见性问题
  5. 指令重排序问题

想要解决线程安全问题就需要从上面这几点出发,由于我们上述的代码不涉及4和5所以无需考虑它们,而第一点是系统原因是客观存在的无法更改。

我们此时有两种解决方法:

  • 将这个代码改为单线程(解决多个线程针对同一个变量进行修改的问题);
  • 让该自增操作变为原子的(解决修改操作不是原子的问题)

这两种方法都可以解决此代码的线程安全问题,第一种很好实现,那么我们该怎样让这个自增操作变为原子的呢?加锁!

synchronized 关键字

synchronized 关键字是JAVA提供的一种常用的加锁工具。

注:

  • synchronized关键字在使用时需要搭配()和{};
  • 程序执行进入 { 加锁 离开 } 解锁 ,{} 里面就是被加锁的代码块;
  • ()里面用来表示一个加锁的对象(这个对象是啥不重要,它的主要功能就是用来区分多个线程是否在竞争同一个锁)。

如果多个线程对同一个线程尝试进行加锁操作就会产生锁竞争(其中一个线程就会发生阻塞等待),如果是不同对象就不会产生锁竞争,仍然是并发执行。
我们先随便创建一个Object类型的对象,命名为lock,将count++放入{}中

  1. private static int count;
  2. public static void main(String[] args) throws InterruptedException {
  3. Object lock = new Object();
  4. Thread t1 = new Thread(()->{
  5. for (int i = 0; i < 10000; i++) {
  6. synchronized(lock) {
  7. count++;
  8. }
  9. }
  10. });
  11. Thread t2 = new Thread(()->{
  12. for (int i = 0; i < 10000; i++) {
  13. synchronized(lock) {
  14. count++;
  15. }
  16. }
  17. });
  18. t1.start();
  19. t2.start();
  20. t1.join();
  21. t2.join();
  22. System.out.println(count);
  23. }

在这里插入图片描述
由于我们对count++加了锁所以线程t1和t2就会在执行过程中相互影响。
当t1线程在执行++操作时,如果t2线程也想执行++操作就会发生阻塞等待,当t1线程执行完++操作出了 } 后会解锁,此时 t2 才会继续向下执行。
在这里插入图片描述

此时这个程序的执行顺序就只会是这类正确的类型:
在这里插入图片描述

发表评论

表情:
评论列表 (有 0 条评论,100人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Java线安全

    2.1 线程安全 如果有多个线程在同时运行,而这些线程可能会同时运行这段代码。程序每次运行结果和单线程运行的结果是一样 的,而且其他的变量的值也和预期的是一样的,就

    相关 Java 线安全

    一、什么是线程安全 当多个线程访问共享资源时,每个线程都会各自对共享资源进程操作,导致数据不一致,造成程序不能正确的得到结果,此时需要让多个线程排队访问共享资源,让线程安

    相关 线安全JAVA

    线程安全对于我们编写多线程代码是非常重要的。 什么是线程安全? 在我们平时的代码中有些代码在单线程程序中可以正常执行,但如果同样的代码放在在多个线程中执行就会引发BUG

    相关 Java线安全

    Java 线程安全 什么是线程安全? 当多个线程访问某个类时,这个类始终都能表现出正确的行为,那么就称这个类是线程安全的。 一:基本概念 共享资源:能够被多个线程

    相关 Java线安全和非线安全

    线程安全就是多线程访问时,采用了加锁机制,当一个线程访问该类的某个数据时,进行保护,其他线程不能进行访问直到该线程读取完,其他线程才可使用。不会出现数据不一致或者数据污染。