Java同步方法:synchronized到底锁住了谁?
前言
相信不少同学在上完Java课后,对于线程同步部分的实战,都会感到不知其然。
比如上课做实验的时候,按着老师的实验指导书中的描述完成了多线程的同步操作,就感觉自己已经掌握这个知识点了,实际运用中再次手足无措,就像我一样。 这里提问一下:synchronized对方法修饰,在别处调用这个方法时,谁被锁定了呢?另外,在新建线程中使用synchronized(this){ }结构时,如:
void methodA() {
new Thread(() -> {
synchronized (this) {
this.methodB();
}
}).start();
}
这个被锁的this又是谁呢?
这篇博文来详细介绍一下线程同步中涉及synchronized修饰的两种用法:同步方法和同步代码块。
才不会说这篇是我对一个项目代码中的线程同步机制感到迷惑而搜资料写的笔记(
同步方法
先开始介绍synchronized修饰符本身的特性:
- synchronized关键字不能被继承 即父类方法是同步方法 子类方法继承后默认不是同步方法
- synchronized不能修饰接口方法 因为接口是特殊的抽象类 不能新建实例 实例锁应归实现其的类所有
- synchronized不能修饰构造方法(但可在内部使用synchronized代码块来同步
类的成员方法
修饰一个普通方法时,作用域是当前调用对象,即只要还没出方法的作用域,其他试图获取该对象的锁线程都将被阻塞。
这里容易误解的就是,只是尝试获取该对象锁的线程会被阻塞,并不影响其他线程不获取锁瞎操作,所以要在涉及同步量操作的所有地方采用同步方法(如加锁),否则引起线程安全问题几乎是必然的。
类的静态方法
因为类的静态方法属于类,而不属于类的某个特定实例,所以对类的静态方法修饰直接作用于类本身,相当于synchronized(ClassA.class),即直接锁定整个类。这里有不少别人的笔记写着,直接作用于类的所有对象,我觉得存在歧义,因为正常情况下,除非采用工厂模式之类的方法,不然很难获取到所有对象的引用,并且这种表述也是不符合直觉的。
同步代码块
由于同步是一个高开销操作,上面讲的同步方法其实是同步代码块的一个语法糖,平时应尽量使用synchronized同步关键代码,而不是对整个方法同步,要尽可能减少同步的内容。
对成员方法修饰 -> synchronized(this)
对静态方法修饰 -> synchronized(ClassA.class)
总结
自己全部测试了一遍,重新验证了猜想,目测没有什么不符合直觉的地方,另外,对单独信号量,如byte[]之类的加锁操作,如果不释放锁,其他线程会全部阻塞在获取锁的过程中,这里不单列出来。
本文前言中提到的问题,答案即为新建这个线程的实例本身,而不是这个被新建的线程类。
这里看到结果就容易理解了,每个对象都自己与一个锁相关联,类静态本身也与一个锁关联,任何尝试获取锁的方法才可能会引起阻塞。
| 修饰对象/其他线程 | 同实例 | 其他实例 | 类 | ||||||
|---|---|---|---|---|---|---|---|---|---|
| 阻塞/不阻塞 | 成员变量 | 非同步方法 | 同步方法 | 成员变量 | 非同步方法 | 同步方法 | 静态变量 | 静态非同步方法 | 静态同步方法 |
| this | 不阻塞 | 不阻塞 | 阻塞 | 不阻塞 | 不阻塞 | 不阻塞 | |||
| 类的成员方法 | 不阻塞 | 不阻塞 | 阻塞 | 不阻塞 | |||||
| 类.class | - | 阻塞 | |||||||
| 类的静态方法 | 阻塞 |
其他同步方法
这里就不多介绍了,下面遇到了再详细写。
- 使用volatile修饰域 每次使用此域都需重新计算
- 使用ReentrantLock可重入锁 需要注意及时手动释放 通常在finally里释放
- 使用ThreadLocal 这里反对本文参考资料中的一个介绍 严格来说这不叫同步 只是各个使用到相同类的线程 独立的创建一份自己的副本 由于这个副本仅当前线程可达 也就没有了其他线程的竞争 相当于线程内部的全局变量 应用场景主要有两种 一是单个线程中多个类的实例共享另一个实例的时候 如数据库连接、RequestContextHolder、Web Session、日志的MDC、SimpleDateFormat(线程不安全的工具类)等 二是避免超长参数传递链 避免在方法中来回传递参数
- 使用LinkedBlockingQueue阻塞队列 利用队列FIFO(先进先出)的特性实现生产者-消费者模型
- 使用Atomic原子变量 利用原子操作本身的特性实现多线程同步
参考资料
java-synchronized-method-lock-on-object-or-method - stackoverflow
what-is-the-reason-why-synchronized-is-not-allowed-in-java-8-interface-methods - stackoverflow
Java线程同步的7种方式 - cnblogs
关于Java的构造方法在类初始化和类实例化中的实质 - CSDN
Java中Synchronized的用法 - CSDN
Java多线程安全之构造函数 - CSDN
正确理解Thread Local的原理与适用场景 - 个人博客
理解Java中的ThreadLocal - 个人博客
Java中的四种引用类型(强、软、弱、虚) - 简书
Java同步方法:synchronized到底锁住了谁?的更多相关文章
- synchronized到底锁住的是谁?
本文代码仓库:https://github.com/yu-linfeng/BlogRepositories/tree/master/repositories/sync 先来一道校招级并发编程笔试题 题 ...
- Java synchronized到底锁住的是什么?
使用环境:多线程java程序中. 作用:在多线程的环境下,控制synchronized代码段不被多个线程同时执行.synchronized既可以加在一段代码上,也可以加在方法上. 使用:synchro ...
- java synchronized究竟锁住的是什么
刚学java的时候,仅仅知道synchronized一个线程锁.能够锁住代码,可是它真的能像我想的那样,能够锁住代码吗? 在讨论之前先看一下项目中常见关于synchronized的使用方法: publ ...
- Java synchronized(this)锁住的是什么
synchronized锁住的是括号里面的对象,而不是代码. 对于非static的synchronized方法,锁的就是对象本身,也就是this.
- 关于synchronized无法锁住Integer原因
原因 在多线程的时候,为了保证数据安全,必须在修改数据时使用线程同步,java中的synchronized用来实现线程同步.线程列队. 学完多线程基础的我,写一个多线程交替输出1,2,3,4,5... ...
- 关于Synchronized关键字锁住对象的嵌套问题
如果在子关键字代码块中调用了sleep,是否会保留有所的锁?
- Java并发,synchronized锁住的内容
synchronized用在方法上锁住的是什么? 锁住的是当前对象的当前方法,会使得其他线程访问该对象的synchronized方法或者代码块阻塞,但并不会阻塞非synchronized方法. 脏读 ...
- 由Java 15废弃偏向锁,谈谈Java Synchronized 的锁机制
Java 15 废弃偏向锁 JDK 15已经在2020年9月15日发布,详情见 JDK 15 官方计划.其中有一项更新是废弃偏向锁,官方的详细说明在:JEP 374: Disable and Depr ...
- 从分布式锁角度理解Java的synchronized关键字
分布式锁 分布式锁就以zookeeper为例,zookeeper是一个分布式系统的协调器,我们将其理解为一个文件系统,可以在zookeeper服务器中创建或删除文件夹或文件.设D为一个数据系统,不具备 ...
随机推荐
- Spring 事务注意事项
使用事务注意事项 1,事务是程序运行如果没有错误,会自动提交事物,如果程序运行发生异常,则会自动回滚. 如果使用了try捕获异常时.一定要在catch里面手动回滚. 事务手动回滚代码 Transact ...
- 数据分析_numpy_基础2
数据分析_numpy_基础2 sqrt 开方 arr = np.arange(10) arr # array([0, 1, 2, 3, 4, 5, 6, 7, 8, 9]) np.sqrt(arr) ...
- 一文彻底读懂MySQL事务的四大隔离级别
前言 之前分析一个死锁问题,发现自己对数据库隔离级别理解还不够清楚,所以趁着这几天假期,整理一下MySQL事务的四大隔离级别相关知识,希望对大家有帮助~ 事务 什么是事务? 事务,由一个有限的数据库操 ...
- c语言提取浮点型数据的整数部分与小数部分几种方法
一 前记 最近涉及到把各种传感器的数据通过wifi和bt传输出去,这就涉及到了浮点传输的问题,为了方便传输,笔者的做法一般是把小数和整数部分分开,分别传输,这就比较简单明晰了. 二 方法论 其实,把浮 ...
- JS 剑指Offer(六) 用两个栈实现队列
题目:用两个栈实现队列,实现它的两个函数appendTail和deleteHead,分别完成在队列尾部插入整数和在队列头部删除整数,若队列中没有元素deleteHead返回-1 分析:在队列的尾部插入 ...
- Git版本控制 Git、github,gitlab相关操作
目录 关于版本控制 版本管理工具 集中式管理 分布式管理 git版本管理 git介绍 软件安装 Git工作状态 原理流程步骤 git基本操作 对文件进行修改 分支 共享仓库 创建共享仓库: 共享仓库上 ...
- Unity 游戏框架搭建 2019 (二十五) 类的第一个作用 与 Obselete 属性
在上一篇我们整理到了第七个示例,我们今天再接着往下整理.我们来看第八个示例: #if UNITY_EDITOR using UnityEditor; #endif using UnityEngine; ...
- 1063 Set Similarity (25分)
Given two sets of integers, the similarity of the sets is defined to be /, where Nc is the number ...
- PTA数据结构与算法题目集(中文) 7-11
PTA数据结构与算法题目集(中文) 7-11 7-11 关键活动 (30 分) 假定一个工程项目由一组子任务构成,子任务之间有的可以并行执行,有的必须在完成了其它一些子任务后才能执行.“任务调度 ...
- thinkphp后端开发ajax接口开发测试
数据自动填充,Firefox的网络查看JSON数据,查错. 同时,用getLastSql函数查询,并且开启日志文件记录网页执行的全过程,还可以用thinkPHP内置的trace的跟踪. 谁有更好地方法 ...