内置锁(一)synchronized 介绍与用法
一、synchronized 的介绍
synchronized 是 Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码,而这段代码也被称为临界区。
synchronized 有多个叫法,而每个叫法都表明synchronized 的特性:
1、内置锁(又叫 隐式锁):synchronized 是内置于JDK中的,底层实现是native;同时,加锁、解锁都是JDK自动完成,不需要用户显式地控制,非常方便。
2、同步锁:synchronized 用于同步线程,使线程互斥地访问某段代码块、方法。这就是意味着最多只有一个线程能够获得该锁,当线程A尝试去获得线程B持有的内置锁时,线程A必须等待或者阻塞,知道线程B释放这个锁,如果B线程不释放这个锁,那么A线程将永远等待下去。
3、对象锁:准确来说,是分为对象锁、类锁。synchronized 以当前的某个对象为锁,线程必须通过互斥竞争拿到这把对象锁,从而才能访问 临界区的代码,访问结束后,就会释放锁,下一个线程才有可能获取锁来访问临界区(被锁住的代码区域)。synchronized锁 根据锁的范围分为 对象锁 和 类锁。对象锁,是以对象实例为锁,当多个线程共享访问这个对象实例下的临界区,只需要竞争这个对象实例便可,不同对象实例下的临界区是不用互斥访问;而类锁,则是以类的class对象为锁,这个锁下的临界区,所有线程都必须互斥访问,尽管是使用了不同的对象实例;
总的来说,对象锁的粒度要比类锁的粒度要细,引起线程竞争锁的情况比类锁要少的多,所以尽量别用类锁,锁的粒度越少越好。
看下面的例子:
FruitCount fruitCount = new FruitCount();
FruitCount fruitCount_3 = new FruitCount();
//线程1、2 使用了同一个FruitCount对象(fruitCount )
Thread thread_1 = new Thread(new MyRunable(fruitCount));
Thread thread_2 = new Thread(new MyRunable(fruitCount));
//线程3使用了不同的FruitCount对象(fruitCount_3 )
Thread thread_3 = new Thread(new MyRunable(fruitCount_3));
线程1、2将会互斥访问getAmount( )方法,线程3则独享getAmount( )方法;线程1、2的getAmount( )方法中的对象锁是fruitCount ,线程3的则是 fruitCount_3;这便是对象锁的粒度范围,不同的对象,锁是相互隔离的。而对于setData( )方法,三个线程都要互斥访问访问它,因为是同一个锁 -- FruitCount.class类锁。
class FruitCount{
static int price = 5;
static int num = 10;
public void setData(int price,int num){
//类锁,以FruitCount.class为锁
synchronized(FruitCount.class){
this.price = price;
this.num = num;
}
}
public int getAmount(){
//对象锁,以当前对象为锁
synchronized (this) {
int amount = price*num;
return amount;
}
}
}
class MyRunable implements Runnable{
FruitCount fruitCount;
public MyRunable(FruitCount fruitCount){
this.fruitCount = fruitCount;
}
@Override
public void run() {
//setData方法 有类锁
fruitCount.setData(5, 10);
//getAmount方法 里面有对象锁,就是fruitCount对象
fruitCount.getAmount();
}
}
二、synchronized 用法
synchronized 的用法只有两种:修饰方法、修饰代码块
1、在方法声明时使用,修饰方法
注意:这个synchronized 修饰符 不会参与方法签名的比较;
语法:
public synchronized void synMethod() {
//方法体
}
有以下两种情况:
- 1.1、修饰的方法是普通的成员方法,那么是对象锁,便是以当前对象为锁,即调用这个方法的对象
- 1.2、修饰的方法是静态方法,则是类锁。
public synchronized static int countData(int data){
return data*data;
}
2、修饰一个代码块
语法:
public int synMethod(int a1){
synchronized( object ) {
//代码块,一次只能有一个线程进入
}
}
有以下3种情况:
2.1、object 是 this,是对象锁,this指代当前对象
public int getAmount(){
//对象锁,以当前对象为锁
synchronized (this) {
int amount = price*num;
return amount;
}
}
2.2、object 是一个普通对象实例
- 如果是静态对象,那么就是 类锁;
- 如果是非静态对象:成员对象变量、局部变量(甚至可以是 方法参数),那么就是对象锁;
public void setObj(){
FruitCount fruitCount = null;
//局部变量
synchronized(fruitCount){
fruitCount = new FruitCount();
}
}
2.3、object 是一个类的class 对象,那么就是类锁
public void setData(int price,int num){
//类锁,以FruitCount.class为锁
synchronized(FruitCount.class){
this.price = price;
this.num = num;
}
}
小 结:
1、出现类锁的情况:
- 以 类.class 为锁
- 以 静态变量为锁
- 修饰静态方法
2、出现对象锁的情况:
- 以实例成员对象为锁(特殊:this 指当前对象)
- 以局部变量(甚至是方法传进来的参数)为锁、
- 修饰成员方法
3、当synchronized修饰方法时,synchronized是不参与 方法签名的比较;
内置锁(一)synchronized 介绍与用法的更多相关文章
- 深入理解Java内置锁和显式锁
synchronized and Reentrantlock 多线程编程中,当代码需要同步时我们会用到锁.Java为我们提供了内置锁(synchronized)和显式锁(ReentrantLock)两 ...
- 七 内置锁 wait notify notifyall; 显示锁 ReentrantLock
Object中对内置锁进行操作的一些方法: Java内置锁通过synchronized关键字使用,使用其修饰方法或者代码块,就能保证方法或者代码块以同步方式执行. 内置锁使用起来非常方便,不需要显式的 ...
- Java 并发:内置锁 Synchronized
摘要: 在多线程编程中,线程安全问题是一个最为关键的问题,其核心概念就在于正确性,即当多个线程訪问某一共享.可变数据时,始终都不会导致数据破坏以及其它不该出现的结果. 而全部的并发模式在解决问题时,採 ...
- 深入理解java内置锁(synchronized)和显式锁(ReentrantLock)
多线程编程中,当代码需要同步时我们会用到锁.Java为我们提供了内置锁(synchronized)和显式锁(ReentrantLock)两种同步方式.显式锁是JDK1.5引入的,这两种锁有什么异同呢? ...
- synchronized内置锁
synchronized内置锁,如果发生阻塞,无法被中断,除非关闭jvm.因此不能从死锁中恢复.
- java synchronized内置锁的可重入性和分析总结
最近在读<<Java并发编程实践>>,在第二章中线程安全中降到线程锁的重进入(Reentrancy) 当一个线程请求其它的线程已经占有的锁时,请求线程将被阻塞.然而内部锁是可重 ...
- Java内置锁synchronized的实现原理
简述Java中每个对象都可以用来实现一个同步的锁,这些锁被称为内置锁(Intrinsic Lock)或监视器锁(Monitor Lock). 具体表现形式如下: 1.普通同步方法,锁的是当前实例对象 ...
- 内置锁(二)synchronized下的等待通知机制
一.等待/通知机制的简介 线程之间的协作: 为了完成某个任务,线程之间需要进行协作,采取的方式:中断.互斥,以及互斥上面的线程的挂起.唤醒:如:生成者--消费者模式.或者某个动作完成,可以唤醒下一 ...
- jvm内置锁synchronized不能被中断
很久没看技术书籍了,今天看了一下<七周七并发模型>前面两章讲的java,写的还是有深度的.看到了一个有demo,说jvm内置锁synchronized是不能被中断的.照着书上写了个demo ...
随机推荐
- Size类型如何调用和定义
cv::Size sz = cv::Size(PROB_W, PROB_H);//Size(srcimage.cols, srcimage.rows) groundtoimage(xylimit, u ...
- [LeetCode&Python] Problem 832. Flipping an Image
Given a binary matrix A, we want to flip the image horizontally, then invert it, and return the resu ...
- 推荐六个在线生成网址二维码的API接口
现在很多大网站都有这样的一个功能,使用手机扫描一下网页上的二维码便可快速在手机上访问网站.想要实现这样的功能其实很简单,下面麦布分享几个在线生成网址二维码的API接口.都是采用http协议接口,无需下 ...
- Stones 优先队列
Because of the wrong status of the bicycle, Sempr begin to walk east to west every morning and walk ...
- 前端内容安全策略(csp)
什么是CSP CSP全称Content Security Policy ,可以直接翻译为内容安全策略,说白了,就是为了页面内容安全而制定的一系列防护策略. 通过CSP所约束的的规责指定可信的内容来源( ...
- 用户态监控网络接口up、down事件
网上搜索(https://blog.csdn.net/qq123386926/article/details/50695725)可以直接使用netlink现成的接口实现: #include <s ...
- 使用VBS发邮件
NameSpace = "http://schemas.microsoft.com/cdo/configuration/"set Email = CreateObject(&quo ...
- 使用python把html网页转成pdf文件
我们看到一些比较写的比较好文章或者博客的时候,想保存下来到本地当一个pdf文件,当做自己的知识储备,以后即使这个博客或者文章的连接不存在了,或者被删掉,咱们自己也还有. 当然咱们作为一个coder,这 ...
- 和为 s 的两个数字(和为 s 的连续正数序列)
题目 输入一个递增排序的数组和一个数字 s,在数组中查找两个数,得它们的和正好是 s.如果有多对数字的和等于 s,输出任意一对即可 思路 我们先在数组中选择两个数字,如果它们的和等于输入的 s,我们就 ...
- php 图片剪切
<?php /** * 图像裁剪 * @param $source_path 原图路径 * @param $target_width 需要裁剪的宽 * @param $target_height ...