多线程基础——Synchronize

简介: 在Java中,`synchronized`关键字用于保证线程安全,通过加锁机制确保多线程环境下的原子性。它可以使线程由并行变为串行,虽然可能影响性能,但确保了结果的正确性。使用时需注意:1. **修饰方法**:锁住实例对象。2. **修饰代码块**:锁住当前调用方法的对象。3. **修饰静态方法**:锁住类对象。`synchronized`特性包括互斥、内存可见性和可重入。互斥确保同一时间只有一个线程能执行锁定代码;内存可见性保证线程间数据的一致性;可重入允许同一线程多次获取同一锁而不会死锁。

线程安全——Synchronized

[toc]

前面我们介绍了在Java中可以用加锁关键字synchronized保证原子性。

线程安全中通过synchronized给线程加锁,是线程由并行变为串行,这时可能会有疑问说:使用多线程的目的就是提高代码效率,加锁后就变成了单线程了,岂不是多此一举了?

因此在使用多线程时要注意一下几点

  1. 使用多线程的前提是必须保证结果的正确
  2. 在多线程修改共享变量时,才会出现线程安全问题;通过缩小锁的范围,从而提高程序的并发处理能力

1.使用方法

1.1修饰方法

这里我们以 :使用多线程实现count自增 为例

java

代码解读

复制代码

class Counter{
    public int count =0;
    public synchronized void increase(){
        count++;
    }
}

1.2修饰代码块

java

代码解读

复制代码

class Counter{
    public int count =0;
    public  void increase(){
        System.out.println("加锁之前的操作...");
        //锁对象 需提供一个对象
        synchronized (this){
            count++;
        }
        System.out.println("加锁之后的操作...");

    }
}

使用synchronized修饰的代码块涉及的指令,并不是在CPU上无脑的执行完,而是有可能执行了一半就被调度走了,但是即便被调度出CPU也不会释放锁,别的线程尝试获取锁时依然会阻塞等待

1.3锁静态方法

java

代码解读

复制代码

class Counter {
    public static int count = 0;
    public synchronized static void increase() {
        count++;
    }
}

1.3 给一个线程加锁,也会出现线程安全问题

举栗:

java

代码解读

复制代码

class Counter{
    public int count =0;

    /**
     * 执行自增
     */
    public synchronized void increase(){
            count++;
    }

    /**
     * 不加锁
     */
    public  void increase1(){
        count++;
    }
}

1.4锁对象

在1.2中**()里面的this指的是:是对那个对象进行加锁**

加锁操作,是针对 一个对象进行的。

而我们需要的理解的是:synchronized锁的是什么,当两个线程竞争的是同一个锁时,才会产生线程安全(两个线程竞争不同的锁时,根本不会产生任何的线程安全,两把锁之间没有任何关系,不会产生阻塞等待)。

因此在判断是否会产生锁竞争时,只需判断一下锁对象是否是同一个对象。

1.5 锁信息的记录

在Java虚拟机中,对象在内存结中的结构可以划分为4个区域:

  • markword:对象头,锁信息,GC,程序计数器
  • 类型指针(_class) :类型
  • 实例数据(instance_data) :成员变量信息
  • 对齐填充(padding)

总结:

synchronized的几种用法

  1. 修饰普通方法,相当于锁实例对象
  2. 对代码块进行加锁,相当于锁当前调用方法的对象(也是实例对象)
  3. 对静态方法进行加锁,相当于锁类对象

2.特性

互斥

一个线程获取了锁之后,其他线程就必须阻塞等待,等到当前获取锁的线程释放了锁之后再去竞争

内存可见(刷新内存)

synchronized实现内存可见的机制是通过原子性完成的,一个线程完成了读、改、写回内存这一操作之后,再释放锁,下一个线程才能获取锁,再开始读取

所以保证了下一个线程读到的值是上一个线程写回主内存的最新值

可重入

在方法调用的链路中,可能存在多个被synchronized修饰的方法或者代码块

对于同一对象,针对同一个线程(当前获得锁的线程)互斥那么这个锁就不可以重入,会产生一个互斥等待的现象,也叫死锁

对于同一锁对象,针对同一个线程(当前获得锁的线程)不互斥那么这个锁就可以重入


转载来源:https://juejin.cn/post/7299621996249694271

相关文章
|
Java 调度
【多线程和高并发】一:线程实现和synchronize
【多线程和高并发】一:线程实现和synchronize
多线程编程核心技术-对象及变量的并发访问-synchronize同步方法(2)(下)
多线程编程核心技术-对象及变量的并发访问-synchronize同步方法(2)(下)
190 0
多线程编程核心技术-对象及变量的并发访问-synchronize同步方法(2)(下)
|
安全
多线程编程核心技术-对象及变量的并发访问-synchronize同步方法(2)(上)
多线程编程核心技术-对象及变量的并发访问-synchronize同步方法(2)(上)
221 0
多线程编程核心技术-对象及变量的并发访问-synchronize同步方法(2)(上)
|
缓存 Java
理论:第八章:线程是什么,有几种实现方式,它们之间的区别是什么,线程池实现原理,JUC并发包,ThreadLocal与Lock和Synchronize区别
理论:第八章:线程是什么,有几种实现方式,它们之间的区别是什么,线程池实现原理,JUC并发包,ThreadLocal与Lock和Synchronize区别
227 0
|
Java
java多线程基础(synchronize关键字)
基础知识 多线程实现方法 使用Thread创建线并启动线程 使用Runnable创建并启动线程 使用内部类创建线程 线程的方法 线程优先级 守护线程 sleep方法 yield方法 join方法 线程同步 基础知识 线程:进程(process)就是一块包含了某些资源的内存区域。
1148 0
|
9月前
|
Java API 微服务
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
425 83
|
6月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
276 6
|
11月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
414 0
|
7月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
502 16
|
6月前
|
Java 调度 数据库
Python threading模块:多线程编程的实战指南
本文深入讲解Python多线程编程,涵盖threading模块的核心用法:线程创建、生命周期、同步机制(锁、信号量、条件变量)、线程通信(队列)、守护线程与线程池应用。结合实战案例,如多线程下载器,帮助开发者提升程序并发性能,适用于I/O密集型任务处理。
677 0

热门文章

最新文章