JAVA synchronized关键字锁机制(中)
synchronized 锁机制简单的用法,高效的执行效率使成为解决线程安全的首选。 下面总结其特性以及使用技巧,加深对其理解。
特性:
1. Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。
2. 当一个线程同时访问object的一个synchronized(this)同步代码块时,其它线程仍然可以访问非修饰的方法或代码块。
3. 当多个线程同时访问object的synchronized(this)同步代码块时,会存在互斥访问,其它线程会阻塞直到获取锁。
4. 当线程访问object的synchronized(this)同步代码块时,同一个线程可以多次获取锁,当然也不需要释放多次。获取和释放必须相同。
5. 所有的对象都可以获取锁,也可以释放锁。
6. 所有的类也可以获取锁和释放锁,因此静态方法也可以加锁。而特性同上。
猜想:
在jvm中对每个对象都有一个记录锁的状态,当同一个线程访问锁时候就会累加,其它线程访问要等到状态变为未锁状态,当让相同线程释放锁会累减。
质疑:
那么对于类锁来说,应该是所有对象都可以获取锁,那么锁是全局的对所有对象都有效?
public class SynchronizedMtdTest { public static void main(String[] args) { new Thread(new Runnable() {
@Override
public void run() {
SynchronizedMtdTest.appendStr();
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
SynchronizedMtdTest.printStr();
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
SynchronizedMtdTest synchronizedMtdTest = new SynchronizedMtdTest();
synchronizedMtdTest.appendStr();
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
SynchronizedMtdTest synchronizedMtdTest = new SynchronizedMtdTest();
synchronizedMtdTest.printStr();
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
SynchronizedMtdTest synchronizedMtdTest = new SynchronizedMtdTest();
synchronizedMtdTest.append();
}
}).start();
} public synchronized static void appendStr() {
System.out.println("pid=" + Thread.currentThread().getId() + "------appendStr------");
try {
Thread.sleep(1000);
System.out.println("pid=" + Thread.currentThread().getId() + "------appendStr1000------");
} catch (InterruptedException e) {
e.printStackTrace();
} } public static void append() {
System.out.println("pid=" + Thread.currentThread().getId() + "------append------");
try {
Thread.sleep(3000);
System.out.println("pid=" + Thread.currentThread().getId() + "------append3000------");
} catch (InterruptedException e) {
e.printStackTrace();
} } public synchronized static void printStr() {
System.out.println("pid=" + Thread.currentThread().getId() + "------printStr------");
try {
Thread.sleep(6000);
System.out.println("pid=" + Thread.currentThread().getId() + "------printStr6000------");
} catch (InterruptedException e) {
e.printStackTrace();
}
} }
结果:
pid=10------appendStr------
pid=14------append------
pid=10------appendStr1000------
pid=13------printStr------
pid=14------append3000------
pid=13------printStr6000------
pid=12------appendStr------
pid=12------appendStr1000------
pid=11------printStr------
pid=11------printStr6000------
分析结果可以看出对于静态方法加锁,所有的线程调用方法,不管怎样都会互斥,而为加锁不会互斥。
因此:
对于类锁来说应该在持久代也就是方法区有对具体类也有加锁机制,而且原理同对象锁。
那么:
可见,上面可以做如下修改达到相同效果。
public static void appendStr() {
synchronized (SynchronizedMtdTest.class) {
System.out.println("pid=" + Thread.currentThread().getId() + "------appendStr------");
try {
Thread.sleep(1000);
System.out.println("pid=" + Thread.currentThread().getId() + "------appendStr1000------");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
} public static void printStr() {
synchronized (SynchronizedMtdTest.class) {
System.out.println("pid=" + Thread.currentThread().getId() + "------printStr------");
try {
Thread.sleep(6000);
System.out.println("pid=" + Thread.currentThread().getId() + "------printStr6000------");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
结果:
pid=10------appendStr------
pid=10------appendStr1000------
pid=13------printStr------
pid=13------printStr6000------
pid=12------appendStr------
pid=12------appendStr1000------
pid=11------printStr------
pid=11------printStr6000------
JAVA synchronized关键字锁机制(中)的更多相关文章
- 由Java 15废弃偏向锁,谈谈Java Synchronized 的锁机制
Java 15 废弃偏向锁 JDK 15已经在2020年9月15日发布,详情见 JDK 15 官方计划.其中有一项更新是废弃偏向锁,官方的详细说明在:JEP 374: Disable and Depr ...
- Java Synchronized 关键字
本文内容 Synchronized 关键字 示例 Synchronized 方法 内部锁(Intrinsic Locks)和 Synchronization 参考资料 下载 Demo Synchron ...
- Java synchronized 关键字详解
Java synchronized 关键字详解 前置技能点 进程和线程的概念 线程创建方式 线程的状态状态转换 线程安全的概念 synchronized 关键字的几种用法 修饰非静态成员方法 sync ...
- JAVA多线程与锁机制
JAVA多线程与锁机制 1 关于Synchronized和lock synchronized是Java的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码 ...
- 深入浅出Java并发包—锁机制(三)
接上文<深入浅出Java并发包—锁机制(二)> 由锁衍生的下一个对象是条件变量,这个对象的存在很大程度上是为了解决Object.wait/notify/notifyAll难以使用的问题. ...
- 深入浅出Java并发包—锁机制(二)
接上文<深入浅出Java并发包—锁机制(一) > 2.Sync.FairSync.TryAcquire(公平锁) 我们直接来看代码 protected final boolean tr ...
- java并发之线程同步(synchronized和锁机制)
使用synchronized实现同步方法 使用非依赖属性实现同步 在同步块中使用条件(wait(),notify(),notifyAll()) 使用锁实现同步 使用读写锁实现同步数据访问 修改锁的公平 ...
- [多线程] 线程中的synchronized关键字锁
为什么要用锁? 在多线程中,难免会出现在多个线程中对同一个对象的实例变量或者全局静态变量进行并发访问的情况,如果不做正确的同步处理,那么产生的后果就是"脏读",也就是取到的数据其实 ...
- Java:synchronized关键字引出的多种锁
前言 Java 中的 synchronized关键字可以在多线程环境下用来作为线程安全的同步锁.本文不讨论 synchronized 的具体使用,而是研究下synchronized底层的锁机制,以及这 ...
随机推荐
- Redis连接(二)
Redis 命令 Redis 命令用于在 redis 服务上执行操作. 要在 redis 服务上执行命令需要一个 redis 客户端.Redis 客户端在我们之前下载的的 redis 的安装包中. 语 ...
- WebService 的简单使用
简单介绍 WebService是一种跨语言,跨进程,跨机器的数据交互技术. SOAP:简单对象访问协议,通过XML数据交互的轻量级协议,WebService就是采用的这种协议 WSDL:web服务描述 ...
- CH5E07 划分大理石【多重背包】
5E07 划分大理石 0x5E「动态规划」练习描述有价值分别为1..6的大理石各a[1..6]块,现要将它们分成两部分,使得两部分价值之和相等,问是否可以实现.其中大理石的总数不超过20000. 输入 ...
- kibana 和ES安装配置常见问题解决
1.下载相同版本的kibana和ES: es5.6.5下载地址:https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-5 ...
- talib 中文文档(十四):Math Transform Functions 数学变换
Math Transform Functions ACOS - Vector Trigonometric ACos 函数名:ACOS 名称:acos函数是反余弦函数,三角函数 real = ACOS( ...
- Mirror--程序访问镜像数据库的超时机制
程序在访问有镜像的数据库和无镜像的数据库时,采用的链接超时时间算法不一样,因此会导致在在有镜像的数据库上设置了15 S的超时时间,而实际的超时时间仅为3.6 S,从而导致有镜像的数据库更容易超时. 在 ...
- Linux中的Buffer Cache和Page Cache echo 3 > /proc/sys/vm/drop_caches Slab内存管理机制 SLUB内存管理机制
Linux中的Buffer Cache和Page Cache echo 3 > /proc/sys/vm/drop_caches Slab内存管理机制 SLUB内存管理机制 http://w ...
- C++模拟Http/Https访问web站点
一.概述 1.Http与Https的区别与联系 在OSI参考模型中Http与Https均属于应用层协议.Http即Hypertext Transfer Protocol,超文本传输协议:而Https为 ...
- python+Nginx+uWSGI使用说明
安装环境 Remote: CentOS 7.4 x64 (django.example.com) Python: Python3.6.5 Django: Django 2.0.4 nWSGI: uw ...
- word安装mathtype
1:window版本的mathtype:https://pan.baidu.com/s/1Yn8kPG9Y9nBPGaotFJaL2Q ,密码spwm 2:点击exe安装 (安装到c盘,将不会出 ...