浅析Java中synchronized与static synchronized
synchronized关键字
synchronized是进行同步处理而保证线程安全。在一个方法中,如果是方法内的私有变量,那个这个变量是线程安全的,但是类中的实例变量是可能会出现线程安全问题的,当多个线程对这个实例变量进行修改,就可能会出现结果并不是我们期望的结果。
线程安全问题
下面一段代码就出现了线程安全问题。
本来当username为a的时候,num应该为100,但是由于设置让t1休眠了2秒,导致num被刷新成了200,导致最后输出时a和b的num都是200。
public class Service {
private int num = 0;
public void add(String username) {
try {
if (username.equals("a")) {
num = 100;
Thread.sleep(2000);
} else {
num = 200;
}
System.out.println(username + " " + num);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public class ThreadA extends Thread {
private Service service;
public ThreadA(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.add("a");
}
}
public class ThreadB extends Thread {
private Service service;
public ThreadB(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.add("b");
}
}
public class Test {
public static void main(String[] args) {
Service service = new Service();
ThreadA t1 = new ThreadA(service);
ThreadB t2 = new ThreadB(service);
t1.start();
t2.start();
}
}
运行结果:
synchronized锁
下面给add方法加个synchronized关键字。
可以看到输出正确。
synchronized进行了同步,使得线程按照顺序进行访问,由于线程t1和t2的监视器都是同一个实例,相当于拥有同一个锁对象,所以可以进行同步访问。
public class Service {
private int num = 0;
public synchronized void add(String username) {
try {
if (username.equals("a")) {
num = 100;
Thread.sleep(2000);
} else {
num = 200;
}
System.out.println(username + " " + num);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
输出结果:
synchronized同步代码块
上面的代码中是使用synchronized将整个方法进行上锁,只有当一个方法执行完毕后,另一个线程才可以执行这个方法,这样会导致性能损耗很大。
返回到线程安全问题,目的其实是为了解决对临界资源访问的问题,所以其实只需要将临界资源进行上锁就可以了,其他部分其实是可以异步进行的。
在下面代码中,doSomeTask方法里前半部分没有进行同步,后面使用了同步代码块进行加锁。
从输出结果可以看到,前面部分A、B两个线程是异步进行访问的,后部分是同步进行访问的。
public class Service {
public void doSomeTask(String username) {
for (int i = 0; i < 5; i++) {
System.out.println("没有同步 " + Thread.currentThread().getName() + " " + i);
}
System.out.println();
synchronized (this) {
for (int i = 0; i < 5; i++) {
System.out.println("同步了" + Thread.currentThread().getName() + " " + i);
}
}
}
}
public class ThreadA extends Thread {
private Service service;
public ThreadA(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.doSomeTask("a");
}
}
public class ThreadB extends Thread {
private Service service;
public ThreadB(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.doSomeTask("b");
}
}
public class Test {
public static void main(String[] args) {
Service service = new Service();
ThreadA t1 = new ThreadA(service);
ThreadB t2 = new ThreadB(service);
t1.setName("A");
t2.setName("B");
t1.start();
t2.start();
}
}
运行结果:
所以使用synchronized同步代码块可以将需要加锁的部分进行上锁就行了,这样可以提高性能。
可以使用synchronized同步代码块加锁临界资源,这样就可以避免出现线程安全问题。
在JDK1.8中的ConcurrentHashMap也使用synchronized锁,synchronized锁的性能已经有了很大的提高。
static synchronized
static方法为类方法,那么对static方法加上synchronized锁后呢?加锁的究竟是什么呢?
其实这时候监视器上锁的对象为这个类对象,而不是一个具体的实例对象,就是所有该类的实例访问这个方法都会进行加锁。
从下面实例可以看出,虽然是两个实例四个线程访问该方法,但是还是进行了同步,因为所有实例访问的是同一把锁,也就是Service类的对象锁,只要是监视器锁对象是同一个,那么都是会进行上锁同步的。
public class Service {
public static synchronized void doSomeTask(String username) {
for (int i = 0; i < 5; i++) {
System.out.println("同步了" + Thread.currentThread().getName() + " " + i);
}
}
}
public class ThreadA extends Thread {
private Service service;
public ThreadA(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.doSomeTask("a");
}
}
public class ThreadB extends Thread {
private Service service;
public ThreadB(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.doSomeTask("b");
}
}
public class Test {
public static void main(String[] args) {
Service service = new Service();
Service service1 = new Service();
ThreadA t1 = new ThreadA(service);
ThreadB t2 = new ThreadB(service);
ThreadA t3 = new ThreadA(service1);
ThreadB t4 = new ThreadB(service1);
t1.setName("A");
t2.setName("B");
t3.setName("C");
t4.setName("D");
t1.start();
t2.start();
t3.start();
t4.start();
}
}
运行结果:
使用同步代码块也可以实现上述功能。
在同步代码块中,将监视的锁对象设置为Service.class,代表监视的是这个类锁,所以也对这个类进行加锁同步。
public class Service {
public void doSomeTask(String username) {
synchronized (Service.class) {
for (int i = 0; i < 5; i++) {
System.out.println("同步了" + Thread.currentThread().getName() + " " + i);
}
}
}
}
synchronized与static synchronized一起使用
四个线程访问两个方法,线程A和C访问task1,是一个普通的synchronized方法,线程B和D访问task2,是一个static synchronized方法。
从输出结果中可以看出,A、C进行了同步,B、D也进行同步,但是B和C在一开始是交替出现输出,代表B和C其实没有同步,就证明它们的锁不是同一把锁。
A和C线程获取的实例的对象锁,而B和D线程获取的是这个类的锁。
public class Service {
public synchronized void task1() {
for (int i = 0; i < 5; i++) {
System.out.println("task1 " + Thread.currentThread().getName() + " " + i);
}
}
public static synchronized void task2() {
for (int i = 0; i < 5; i++) {
System.out.println("task2 " + Thread.currentThread().getName() + " " + i);
}
}
}
public class ThreadA extends Thread {
private Service service;
public ThreadA(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.task1();
}
}
public class ThreadB extends Thread {
private Service service;
public ThreadB(Service service) {
this.service = service;
}
@Override
public void run() {
super.run();
service.task2();
}
}
public class Test {
public static void main(String[] args) {
Service service = new Service();
ThreadA t1 = new ThreadA(service);
ThreadB t2 = new ThreadB(service);
ThreadA t3 = new ThreadA(service);
ThreadB t4 = new ThreadB(service);
t1.setName("A");
t2.setName("B");
t3.setName("C");
t4.setName("D");
t1.start();
t2.start();
t3.start();
t4.start();
}
}
运行结果:
综上:
- synchronized同步代码块的性能比synchronized同步方法性能好。
- 线程间的同步主要是看的获取的是什么锁,只有需要获取同一把锁的线程才会进行同步。
- static synchronized获取的是类对象锁,而普通的synchronized获取的是实例对象的锁,所以其实他们不是同一把锁。
- static synchronized的类对象锁可以对所有该类实例进行同步,因为所有该类实例获取的类锁都是同一把锁。
浅析Java中synchronized与static synchronized的更多相关文章
- synchronized与static synchronized 的差别、synchronized在JVM底层的实现原理及Java多线程锁理解
本Blog分为例如以下部分: 第一部分:synchronized与static synchronized 的差别 第二部分:JVM底层又是怎样实现synchronized的 第三部分:Java多线程锁 ...
- Java面试之synchronized 和 static synchronized
面试题: 答案: 不能 不能 不能 不能 能 正文 概述 通过分析这两个用法的分析,我们可以理解java中锁的概念.一个是实例锁(锁在某一个实例对象上,如果该类是单例,那么该锁也具有全局锁的概念), ...
- Java中的ReentrantLock和synchronized两种锁定机制的对比
问题:多个访问线程将需要写入到文件中的数据先保存到一个队列里面,然后由专门的 写出线程负责从队列中取出数据并写入到文件中. http://blog.csdn.net/top_code/article/ ...
- Synchronized和Static Synchronized区别
通过分析这两个用法的分析,我们可以理解Java中锁的概念. 一个是实例锁(锁在某一个实例对象上,如果该类是单例,那么该锁也具有全局锁的概念), 一个是全局锁(该锁针对的是类,无论实例多少个对象,那么线 ...
- [Android Pro] synchronized与static synchronized 的区别
reference to : http://www.cnblogs.com/shipengzhi/articles/2223100.html 1.synchronized与static synchr ...
- synchronized和static synchronized的比较
群里讨论的一个问题,网上别人已经贴出了很详细的说明,这里补充记录下,后面加入个人测试代码. 起因:1月份的时候看群里讨论一道问题,问题内容如下: 一个日本作者-结成浩的<java多线程设计模式& ...
- synchronized与static synchronized 差异
1.synchronized与static synchronized 差异 synchronized是对类的当前实例进行加锁,防止其它线程同一时候訪问该类的该实例的全部synchroniz ...
- 多线程同步锁和死锁以及synchronized与static synchronized 的区别
线程:线程是进程中的一个执行单元,负责当前进程中程序的执行,一个进程中至少有一个线程.一个进程中是可以有多个线程的,这个应用程序也可以称之为多线程程序.简而言之:一个程序运行后至少有一个进程,一个进程 ...
- 浅析Java中的final关键字
浅析Java中的final关键字 谈到final关键字,想必很多人都不陌生,在使用匿名内部类的时候可能会经常用到final关键字.另外,Java中的String类就是一个final类,那么今天我们就来 ...
随机推荐
- iOS 用Swipe手势和动画实现循环播放图片
主要想法 添加3个ImageView展示图片,实现图片的无限循环. 使用Swipe手势识别用户向右或向左滑动图片. 使用CATransition给ImageView.layer添加动画,展示图片更换的 ...
- 卷积神经网络(CNN)模型结构
在前面我们讲述了DNN的模型与前向反向传播算法.而在DNN大类中,卷积神经网络(Convolutional Neural Networks,以下简称CNN)是最为成功的DNN特例之一.CNN广泛的应用 ...
- C#开发step步骤条控件
现在很多的javascript控件,非常的不错,其中step就是一个,如下图所示: 那么如何用C#来实现一个step控件呢? 先定义一个StepEntity类来存储步骤条节点的信息: public c ...
- 页面添加数据的PHP
(接前面写的) 第一个页面tianjia.php <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" ...
- Android 视频编辑 SDK
Android 视频编辑 SDK接入说明 一.名词解释 分辨率:用于计算机视频处理的图像,以水平和垂直方向上所能显示的像素数来表示分辨率.常见视频分辨率的有1080P即1920x1080,720P即1 ...
- Java设计模式之《外观模式》及应用场景
原创作品,可以转载,但是请标注出处地址http://www.cnblogs.com/V1haoge/p/6484128.html 1.外观模式简介 外观模式,一般用在子系统与访问之间,用于对访问屏蔽复 ...
- 用smarty模板做的登录
用smarty模板做的登录和之前我们用php做的登录区别不大 首先要新建一个php文件 一般php文件,要放在这个文件里 它对应的html文件,要放在这个目录里 下面先来做php文件 要先引入入口文件 ...
- 关于mvc中传入DataTable到视图的应用
MVC 中 如果是多表连接查询出的数据需要重新定义一个ViewModel,觉得很是麻烦,所以可以通过传一个DataTable到视图中可以避免这个问题 但是不知道会有什么不好的地方,有这方面经验的大神有 ...
- Intelligent idea高效实用总结
一直使用eclipse,最近才转到idea IDE上面来,的确从效率等多个角度,idea都要优于eclipse.由于刚实用idea,不是很熟练,将常用的技巧总结集锦一下,方便以后查看,慢慢积累吧 一. ...
- koa中间件系统原理及koa+orm2实践。
koa是由 Express 原班人马打造的新的web框架.套用其官方的说法:Koa 应用是一个包含一系列中间件 generator 函数的对象. 这些中间件函数基于 request 请求以一个类似于栈 ...