synchronized的作用
一、同步方法
public synchronized void methodAAA(){ //…. }
锁定的是调用这个同步方法的对象
测试:
a、不使用这个关键字修饰方法,两个线程调用同一个对象的这个方法。
目标类:
public class TestThread { public void execute(){ //synchronized,未修饰 for(int i=0;i<100;i++){ System.out.println(i); } } }
线程类:
public class ThreadA implements Runnable{ TestThread test=null; public ThreadA(TestThread pTest){ //对象有外部引入,这样保证是同一个对象 test=pTest; } public void run() { test.execute(); } }
调用:
TestThread test=new TestThread(); Runnable runabble=new ThreadA(test); Thread a=new Thread(runabble,"A"); a.start(); Thread b=new Thread(runabble,"B"); b.start();
结果:
输出的数字交错在一起。说明不是同步的,两个方法在不同的线程中是异步调用的。
b、修改目标类,增加synchronized修饰
public class TestThread { public synchronized void execute(){ //synchronized修饰 for(int i=0;i<100;i++){ System.out.println(i); } } }
结果:
输出的数字是有序的,首先输出A的数字,然后是B,说明是同步的,虽然是不同的线程,但两个方法是同步调用的。
注意:上面虽然是两个不同的线程,但是是同一个实例对象。下面使用不同的实例对象进行测试。
c、每个线程都有独立的TestThread对象。
目标类:
public class TestThread { public synchronized void execute(){ //synchronized修饰 for(int i=0;i<100;i++){ System.out.println(i); } } }
线程类:
public class ThreadA implements Runnable{ public void run() { TestThread test=new TestThread(); test.execute(); } }
调用:
Runnable runabble=new ThreadA(); Thread a=new Thread(runabble,"A"); a.start(); Thread b=new Thread(runabble,"B"); b.start();
结果:
输出的数字交错在一起。说明虽然增加了synchronized 关键字来修饰方法,但是不同的线程调用各自的对象实例,两个方法仍然是异步的。
引申:
对于这种多个实例,要想实现同步即输出的数字是有序并且按线程先后顺序输出,我们可以增加一个静态变量,对它进行加锁(后面将说明锁定的对象)。
修改目标类:
public class TestThread { private static Object lock=new Object(); //必须是静态的。 public void execute(){ synchronized(lock){ for(int i=0;i<100;i++){ System.out.println(i); } } } }
二、同步代码块
public void method(SomeObject so){ synchronized(so) //….. } }
锁定一个对象,其实锁定的是该对象的引用(object reference)
谁拿到这个锁谁就可以运行它所控制的那段代码。当有一个明确的对象作为锁时,就可以按上面的代码写程序,但当没有明确的对象作为锁,只是想让一段代码同步时,可以创建一个特殊的instance变量(它必须是一个对象)来充当锁(上面的解决方法就是增加了一个状态锁)。
a、锁定一个对象,它不是静态的
private byte[] lock = new byte[0]; // 特殊的instance变量
目标类:
public class TestThread { private Object lock=new Object(); public void execute(){ synchronized(lock){ //增加了个锁,锁定了对象lock,在同一个类实例中,是线程安全的,但不同的实例还是不安全的。因为不同的实例有不同对象锁lock. for(int i=0;i<100;i++){ System.out.println(i); } } } }
其实上面锁定一个方法,等同于下面的:
public void execute(){ synchronized(this){ //同步的是当然对象 for(int i=0;i<100;i++){ System.out.println(i); } } }
b、锁定一个对象或方法,它是静态的
这样锁定,它锁定的是对象所属的类
public synchronized static void execute(){ //... } 等同于
public class TestThread { public static void execute(){ synchronized(TestThread.class){ // } } }
测试:
目标类:
public class TestThread { private static Object lock=new Object(); public synchronized static void execute(){ //同步静态方法 for(int i=0;i<100;i++){ System.out.println(i); } } public static void execute1(){ for(int i=0;i<100;i++){ System.out.println(i); } } public void test(){ execute(); //输出是有序的,说明是同步的 //execute1(); //输出是无须的,说明是异步的 } } 线程类:调用不同的方法,于是建立了两个线程类 public class ThreadA implements Runnable{ public void run() { TestThread.execute();//调用同步静态方法 } } public class ThreadB implements Runnable{ public void run() { TestThread test=new TestThread(); test.test();//调用非同步非静态方法 } } 调用: Runnable runabbleA=new ThreadA(); Thread a=new Thread(runabbleA,"A"); a.start(); Runnable runabbleB=new ThreadB(); Thread b=new Thread(runabbleB,"B"); b.start(); 注意: 用synchronized 来锁定一个对象的时候,如果这个对象在锁定代码段中被修改了,则这个锁也就消失了。看下面的实例: 目标类: public class TestThread { private static final class TestThreadHolder { private static TestThread theSingleton = new TestThread(); public static TestThread getSingleton() { return theSingleton; } private TestThreadHolder() { } } private Vector ve =null; private Object lock=new Object(); private TestThread(){ ve=new Vector(); initialize(); } public static TestThread getInstance(){ return TestThreadHolder.getSingleton(); } private void initialize(){ for(int i=0;i<100;i++){ ve.add(String.valueOf(i)); } } public void reload(){ synchronized(lock){ ve=null; ve=new Vector(); //lock="abc"; for(int i=0;i<100;i++){ ve.add(String.valueOf(i)); } } System.out.println("reload end"); } public boolean checkValid(String str){ synchronized(lock){ System.out.println(ve.size()); return ve.contains(str); } } }
说明:在reload和checkValid方法中都增加了synchronized关键字,对lock对象进行加锁。在不同线程中对同一个对象实例分别调用reload和checkValid方法。
在reload方法中,不修改lock对象即注释lock="abc"; ,结果在控制台输出reload end后才输出100。说明是同步调用的。
如果在reload方法中修改lock对象即去掉注释,结果首先输出了一个数字(当前ve的大小),然后输出reload end。说明是异步调用的。
2、单例模式中对多线程的考虑
public class TestThread { private static final class TestThreadHolder { private static TestThread theSingleton = new TestThread(); public static TestThread getSingleton() { return theSingleton; } private TestThreadHolder() { } } private Vector ve =null; private Object lock=new Object(); private TestThread(){ ve=new Vector(); initialize(); } public static TestThread getInstance(){ return TestThreadHolder.getSingleton(); } }
说明:增加了一个内部类,在内部类中申明一个静态的对象,实例化该单例类,初始化的数据都在单例类的构造函数中进行。这样保证了多个实例同时访问的时候,初始化的数据都已经成功初始化了。
总结:
A. 无论synchronized关键字加在方法上还是对象上,它取得的锁都是对象,而不是把一段代码或函数当作锁,所以首先应知道需要加锁的对象
B.每个对象只有一个锁(lock)与之相关联。
C.实现同步是要很大的系统开销作为代价的,甚至可能造成死锁,所以尽量避免无谓的同步控制。
synchronized的作用的更多相关文章
- JAVA关键词synchronized的作用
记下来,很重要. Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问同一个对象object中的这个synchron ...
- 搜狗一道java题目 关于对象 synchronized 关键字作用在 int, integer
第一次见到这个题目,我觉得自己没学到java,太浅了,其实这个问题没有考synchronized关键字,只是考什么是对象? 1.在java编程思想的第二章有一句话; 一切都是对象,很可惜int,c ...
- Android 虹软2.0人脸识别,注册失败问题 分析synchronized的作用
人脸识别需要init初始化(FaceServer中),离开时需要unInit销毁:当一个含有人脸识别的界面A跳向另一个含有人脸识别的界面B时,由于初始化和销毁都是对FaceServer类加锁(sync ...
- synchronized 的作用?
在 Java 中,synchronized 关键字是用来控制线程同步的,就是在多线程的环境 下,控制 synchronized 代码段不被多个线程同时执行. synchronized 既可以加在一段代 ...
- 从JAVA看C#中volatile和synchronized关键字的作用
最近一直在想C#中 volatile关键字到底是用来干什么的?查了很多.NET的文章都是说用volatile修饰的变量可以让多线程同时修改,这是什么鬼... 然后查到了下面这篇JAVA中关于volat ...
- Java 关键字volatile 与 synchronized 作用与区别
1,volatile 它所修饰的变量不保留拷贝,直接访问主内存中的. 在Java内存模型中,有main memory,每个线程也有自己的memory (例如寄存器).为了性能,一个线程会在自己 ...
- synchronized和lock的作用与对比
一.synchronized的作用 synchronized是java中的一个关键字,用于线程同步.1. 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码,作用的对象 ...
- java中关键字volatile的作用
用在多线程,同步变量. 线程为了提高效率,将某成员变量(如A)拷贝了一份(如B),线程中对A的访问其实访问的是B.只在某些动作时才进行A和B的同步.因此存在A和B不一致的情况.volatile就是用来 ...
- Thread 学习记录 <1> -- volatile和synchronized
恐怕比较一下volatile和synchronized的不同是最容易解释清楚的.volatile是变量修饰符,而synchronized则作用于一段代码或方法:看如下三句get代码: int i1; ...
随机推荐
- 分布式版本控制git常见问题之gitignore冲突(精简版)
上次写的的太模糊了,现在简单直接写出个人心得,如下: 原因是有人提交了.gitignore里面的内容,所以和本地的不一样,这样就有问题,那么pull都不可以,所以要这样: git update-ind ...
- 弹出框插件layer使用
layer是一款近年来备受青睐的web弹层组件,她具备全方位的解决方案,致力于服务各水平段的开发人员,您的页面会轻松地拥有丰富友好的操作体验. 插件官方地址:http://layer.layui.co ...
- idea的一揽子工程
我总觉得,世上如果人人都像我一样,那路该多难走.有时候在网上找到一些问题的答案,成功解决之后,就这么过去了,实在罪过. 将idea的几个常见的使用问题综合到一起吧,如果有不会用的,欢迎留言.当然,请支 ...
- Range Modular Queries
Range Modular Queries 题意 给出一个数列,q个查询,问查询区间内有几个数 a[i] % x == y. 分析 其实裸的分块就能过了,跑的还特别快. 这里分块的作用就是排序. 在x ...
- PHP版本替换, phpinfo和php -v显示版本信息不一致
环境:OS X EI Capitan 10.11 & lnmp 背景: 1想将lamp(xampp安装的,php5.2)换成 lnmp(php7.0) 2php5.2卸载(xampp卸载& ...
- git与github的使用
git和github是两个完全不同的概念,就好比如雷锋与雷峰塔的关系. git是一个版本管理工具,用来更好的管理你的程序,比如你原来提交过的内容,以后虽然修改了,但是通过git这个工具, 可以把你原来 ...
- maven问题:org.springframewor.web.filter.CharacterEncodingFileter不能强转为javax.servlet.Filter
使用maven搭建ssm(SpringMVC.Spring .Mybatis)项目,启动时报: java.lang.ClassCastException:org.springframewor.web. ...
- 非Controller中调用Service
1. 新增文件 package com.library.common; import org.springframework.beans.BeansException; import or ...
- Caused by: java.lang.RuntimeException: by java.lang.OutOfMemoryError: PermGen space(tomcat 启动时提示内存溢出)
设置MaxPermSize大小TOMCAT_HOME/bin/catalina.bat 文件头加set JAVA_OPTS='-Xms512m -Xmx1024m -XX:MaxPermSize=51 ...
- [转] .NET领域驱动设计—初尝(疑问、模式、原则、工具、过程、框架、实践)
阅读目录: 1.1.疑问 1.1.1.UML何用 1.1.2.领域建模 1.2.模式 1.3.原则 1.5.过程 1.6.框架 1.7.项目演示 最近在研究DDD颇有收获,所以整理出来跟大家分享,共同 ...