Java线程和多线程(十四)——Synchronized关键字解析
曾经有一个比较有趣的面试问题,那就是,关于使用synchronized
关键字,是用在方法上面尾号,还是用在一个代码块上面为好?
答案就是使用锁定代码块为更好。因为这样不会锁定对象。当synchronized
关键字在实例方法的上面时,线程对于该方法的访问会直接锁定整个对象,参考如下代码:
class Sync {
public synchronized void syncMethod() {
System.out.println("Sync Method Executed");
}
public void syncThis() {
synchronized (this) {
System.out.println("Sync This Executed");
}
}
}
上面的类中,syncMethod()
和syncThis()
方法,是没有区别的。为了验证这个说法,参考如下代码:
package net.ethanpark.common.task;
/**
* Author: EthanPark <br/>
* Date: 2016/11/27<br/>
*/
public class SyncTest {
public static void main(String[] args) {
Sync sync = new Sync();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
sync.syncMethod();
}
});
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
sync.syncThis();
}
});
t1.start();
t2.start();
}
}
class Sync {
public synchronized void syncMethod() {
System.out.println("Sync Method Start To Executed");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Sync Method Executed");
}
public void syncThis() {
synchronized (this) {
System.out.println("Sync This Executed");
}
}
}
再说上面的代码中,我启动了两个线程,分别执行syncMethod()
和syncThis()
方法,其中syncMethod()
方法会等待5秒,然后才执行结束。以下是输出的结果:
Sync Method Start To Executed
Sync Method Executed
Sync This Executed
t1和t2是先后启动的,如果syncMethod()
方法没有锁定Sync对象的话,syncThis()
方法肯定会更优先的执行,但是syncThis()
方法从输出来看,在获取this
对象的锁的时候,阻塞了,直到syncMethod()
全部执行完毕才开始执行,所以才有如上的输出。
在考虑前面那个问题就清晰了,synchronized
直接加在方法上面,会锁定整个对象,也就是说,如果对象中有一些字段是不需要同时来同步的,synchronized
关键字也会阻止其他线程来获取该对象的锁,从而令其他线程阻塞而无法达到高吞吐量。
当然,synchronized
关键字对对象加锁,也只有其他对象也在请求对象锁的时候才会阻塞的,如果其他对象不需要请求对象锁,而是直接访问非同步变量,还是允许的。针对之前的代码我增加了一个新的线程和一个新的方法notSyncMethod()
:
package net.ethanpark.common.task;
/**
* Author: EthanPark <br/>
* Date: 2016/11/27<br/>
*/
public class SyncTest {
public static void main(String[] args) {
Sync sync = new Sync();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
sync.syncMethod();
}
});
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
sync.syncThis();
}
});
Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
sync.notSyncMethod();
}
});
t1.start();
t2.start();
t3.start();
}
}
class Sync {
public synchronized void syncMethod() {
System.out.println("Sync Method Start To Executed");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Sync Method Executed");
}
public void syncThis() {
synchronized (this) {
System.out.println("Sync This Executed");
}
}
public void notSyncMethod(){
System.out.println("Not Sync Method Executed");
}
}
程序的输出如下:
Sync Method Start To Executed
Not Sync Method Executed
Sync Method Executed
Sync This Executed
从第一行输出来看,t1其实已经获得了sync
对象的对象锁,但是notSyncMethod()
方法仍然是可以执行的。
需要注意的是,synchronized
方法也是可以加在静态方法上面的。然而加在静态方法上面和加在实例方法上面,两者是不会使得线程互斥的。原因很简单,针对实例方法同步,我们请求的是对象锁,而静态方法的访问,是不针对对象的,所以两者是不会冲突的。针对静态方法的同步,实际上跟针对类型的同步是一致的。参考如下代码:
package net.ethanpark.common.task;
/**
* Author: EthanPark <br/>
* Date: 2016/11/27<br/>
*/
public class SyncTest {
public static void main(String[] args) {
Sync sync = new Sync();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
sync.syncMethod();
}
});
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
Sync.staticSyncMethod();
}
});
Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
Sync.staticTypeMethod();
}
});
t1.start();
t2.start();
t3.start();
}
}
class Sync {
public synchronized void syncMethod() {
System.out.println("Sync Method Start To Executed");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Sync Method Executed");
}
public synchronized static void staticSyncMethod(){
System.out.println("Static Sync Method Start To Executed");
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Static Sync Method Executed");
}
public static void staticTypeMethod(){
synchronized (Sync.class){
System.out.println("Static Sync Type Executed");
}
}
}
上面代码的输出如下:
Sync Method Start To Executed
Static Sync Method Start To Executed
Static Sync Method Executed
Static Sync Type Executed
Sync Method Executed
可以看出,在staticSyncMethod()
和staticTypeMethod()
两者是等同的,当staticSyncMethod()
执行的时候,staticTypeMethod()
是阻塞的。而两个静态的方法,跟实例的方法是完全互不影响的。
Java线程和多线程(十四)——Synchronized关键字解析的更多相关文章
- java提高篇(十四)-----关键字final
在程序设计中,我们有时可能希望某些数据是不能够改变的,这个时候final就有用武之地了.final是java的关键字,它所表示的是"这部分是无法修改的".不想被改变的原因有两个:效 ...
- java线程池 多线程搜索文件包含关键字所在的文件路径
文件读取和操作类 import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; publi ...
- Java线程和多线程(四)——主线程中的异常
作为Java的开发者,在运行程序的时候会碰到主线程抛异常的情况.如果开发者使用Java的IDE比如Eclipse或者Intellij IDEA的话,可能是不需要直接面对这个问提的,因为IDE会处理运行 ...
- JAVA之旅(十四)——静态同步函数的锁是class对象,多线程的单例设计模式,死锁,线程中的通讯以及通讯所带来的安全隐患,等待唤醒机制
JAVA之旅(十四)--静态同步函数的锁是class对象,多线程的单例设计模式,死锁,线程中的通讯以及通讯所带来的安全隐患,等待唤醒机制 JAVA之旅,一路有你,加油! 一.静态同步函数的锁是clas ...
- Java中的集合(十四) Map的实现类LinkedHashMap
Java中的集合(十四) Map的实现类LinkedHashMap 一.LinkedHashMap的简介 LinkedHashMap是Map接口的实现类,继承了HashMap,它通过重写父类相关的方法 ...
- Java 设计模式系列(十四)命令模式(Command)
Java 设计模式系列(十四)命令模式(Command) 命令模式把一个请求或者操作封装到一个对象中.命令模式允许系统使用不同的请求把客户端参数化,对请求排队或者记录请求日志,可以提供命令的撤销和恢复 ...
- 程序员Java架构师多线程面试题和回答解析
当我们在Java架构师面试的过程中常见的多线程和并发方面的问题肯定是必不可少的一部分.那么在面试之前我们更应该多准备一些关于多线程方面的问题. 面试官只是想确信面试者有足够的Java线程与并发方面的知 ...
- synchronized 关键字解析
synchronized 关键字解析 同步锁依赖于对象,每个对象都有一个同步锁. 现有一成员变量 Test,当线程 A 调用 Test 的 synchronized 方法,线程 A 获得 Test 的 ...
- Java多线程4:synchronized关键字
原文:http://www.cnblogs.com/skywang12345/p/3479202.html 1. synchronized原理在java中,每一个对象有且仅有一个同步锁.这也意味着,同 ...
随机推荐
- Django之自定义权限
官方解释 Custom permissions¶ To create custom permissions for a given model object, use the permissions ...
- Spring常用注解简单汇总
使用注解之前要开启自动扫描功能,其中base-package为需要扫描的包(含子包). <context:component-scan base-package="cn.test&qu ...
- HTML--<frameset>標簽
<html><frameset rows="20,80"> <frame src="/example/html/frame_a.html&q ...
- 手把手教你制作AppPreview视频并上传到appStore进行审核
手把手教你制作AppPreview视频并上传到appStore进行审核 注意,你需要使用iMovie才能够制作AppPreview视频文件,用QuickTime录制的无效! 最终效果 1. 新建一个事 ...
- [翻译] DraggableYoutubeFloatingVideo
DraggableYoutubeFloatingVideo DraggableYoutubeFloatingVideo allows you to play videos on a floating ...
- [原创]获取JS数组最大值、最小值
核心关键 JS有Array数组对象,使用prototype内置属性扩展,增加Array数组max().min()方法 具体代码 //最小值 Array.prototype.min = function ...
- 基于springMVC的RESTful服务实现
一,什么是RESTful RESTful(RESTful Web Services)一种架构风格,表述性状态转移,它不是一个软件,也不是一个标准,而是一种思想,不依赖于任何通信协议,但是开发时要成功映 ...
- IP地址编址
比特:一比特就是一个数字,1或者0. 字节:以字节是7比特或者8比特,取决于是否使用奇偶校验 八位组:8比特构成 网络地址:用来将数据包发送到远端网路 比如10.0.0.0 广播地址:将信息发送给网络 ...
- taskset
常常感觉系统资源不够用,一台机子上跑了不下3个比较重要的服务,但是每天我们还要在上面进行个备份压缩等处理,网络长时间传输,这在就很影响本就不够用的系统资源: 这个时候我们就可以把一些不太重要的比如co ...
- JavaScript的DOM操作获取元素周边大小
一.clientLeft 和 clientTop 这组属性可以获取元素设置了左边框和上边框的大小,目前只提供了 Left 和 Top 这组,并没有提供 Right 和 Bottom. <scri ...