当前位置: 首页 > article >正文

【Java】synchronized 基础线程安全

欢迎浏览高耳机的博客

希望我们彼此都有更好的收获

感谢三连支持!

在多线程编程中,线程安全是一个至关重要的概念。Java 提供了多种机制来处理线程安全问题,其中 synchronized 关键字是最常用和最基础的一种。本文将介绍线程安全问题的原因,并解释 synchronized 关键字如何帮助解决这些问题。 


线程安全问题通常发生在多个线程同时访问和修改共享数据时,如果没有适当的同步机制,就可能导致数据不一致、程序崩溃或其他不可预测的行为。以下是一些具体的原因:

线程安全问题的原因

  1. 共享资源竞争:当多个线程同时访问和修改同一个资源(如变量、数据结构等),而没有适当的同步措施,就可能产生竞争条件(Race Condition)。

  2. 不可预测的执行顺序:由于线程调度的不确定性,线程的执行顺序可能与程序员的预期不符,导致程序行为异常。

  3. 内存可见性问题:在没有同步的情况下,一个线程对共享变量的修改可能不会立即对其他线程可见,这可能导致其他线程读取到过时的数据。

synchronized 关键字的工作原理

synchronized 是 Java 提供的一种内置的同步机制,它可以用来修饰方法或代码块。当一个线程访问被 synchronized 修饰的方法或代码块时,它会获取一个锁,这个锁通常与对象关联。如果其他线程已经持有该锁,那么它们将被阻塞,直到锁被释放。

synchronized使用实例 

实例 1:同步方法

public class Counter {
    private int count = 0;

    // 同步方法
    public synchronized void increment() {
        count++;
    }

    public synchronized int getCount() {
        return count;
    }

    public static void main(String[] args) {
        Counter counter = new Counter();
        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                counter.increment();
            }
        });

        Thread t2 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                counter.increment();
            }
        });

        t1.start();
        t2.start();

        try {
            t1.join();
            t2.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.out.println("Final count: " + counter.getCount());
    }
}

在这个实例中,increment 方法被 synchronized 关键字修饰,确保了每次只有一个线程可以执行这个方法,从而保证了 count 变量的线程安全。

实例 2:同步静态方法

public class Counter {
    private static int count = 0;

    // 同步静态方法
    public static synchronized void increment() {
        count++;
    }

    public static int getCount() {
        return count;
    }

    public static void main(String[] args) {
        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                Counter.increment();
            }
        });

        Thread t2 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                Counter.increment();
            }
        });

        t1.start();
        t2.start();

        try {
            t1.join();
            t2.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.out.println("Final count: " + Counter.getCount());
    }
}

在这个实例中,increment 方法被声明为 static synchronized,这意味着它是针对类的所有实例共享的 count 变量进行同步的。

synchronized 的解决方案

  1. 互斥锁:确保同一时间只有一个线程可以执行临界区代码。

  2. 内存可见性synchronized 保证了对共享变量的修改对其他线程立即可见。

  3. 原子性:通过锁机制,确保对共享资源的操作是原子性的,即不可分割的。

 

线程安全是多线程编程中的核心问题,synchronized 关键字提供了一种简单有效的同步机制。然而,为了优化性能和适应不同的应用场景,了解和掌握其他线程安全解决方案也是必要的。通过合理设计和使用这些机制,可以确保多线程程序的正确性和性能。 


希望这篇博客能为你理解线程安全以及synchronized关键字提供一些帮助。

如有不足之处请多多指出。

我是高耳机。

 


http://www.kler.cn/news/309101.html

相关文章:

  • git 更新LingDongGui问题解决
  • Reactive 编程-Loom 项目(虚拟线程)
  • 1. YOLOv10: Real-Time End-to-End Object Detection
  • Linux进阶系列(三)——重定向、tee、rsync、xargs
  • Android注册广播
  • 开源免费的NAS系统-TrueNAS CORE搭建和使用(保姆级教程)
  • 基于C++实现(MFC)职工工作量统计系统
  • 机器学习--逻辑回归
  • JavaScript如何判断输入的是空格
  • 常见的反爬虫和应对方法
  • 【SQL】数据库详解-标准SQL语句
  • 协同过滤算法商品推荐系统设计与实现
  • 解决ruoyi-vue-pro-master框架引入报错,启动报错问题
  • 毕设开源 基于python的搜索引擎设计与实现
  • 智能 Uber 发票 PDF 合并工具
  • 【乐企-业务篇】乐企开票具体代码实现
  • Java和西门子S7-1200通讯调试记录
  • GC的算法
  • 从基础到高级:模块化RAG技术全览
  • 【云原生监控】Prometheus之Alertmanager报警
  • 线性基速通
  • 哪款宠物空气净化器是除浮毛王者?希喂、范罗士、霍尼韦尔宠物空气净化器实测
  • 爬坑--docker构建容器ssh连接容器环境变量会发生变化
  • Redis的IO模型
  • 计算机网络分层结构解析:OSI与TCP/IP模型
  • Blender渲染太慢怎么办?blender云渲染已开启
  • 在设计开发中,如何提高网站的用户体验?
  • Qt开发技巧(四)“tr“使用,时间类使用,Qt容器取值,类对象的删除,QPainter画家类,QString的转换,用好 QVariant类型
  • Vite项目中eslint的简单配置
  • Amazon 正式官宣取消居家上班(WFH)