Java多线程同步Synchronized使用分析

同步的概念:

同步分为 同步方法 和 同步块 两种方式。

锁定的内容分为 锁定类的某个特定实例 和 锁定类对象(类的所有实例)

变量分为 实例变量(不带static的变量) 和 类变量(带static的变量)

使用同步的原因

1. 在系统中对访类要使用多线程进行访问;

2. 在该类中有 类变量, 或者是 在类的方法中有访问 公共资源(如一个外部文件的读写)。

同步锁锁定的内容是什么?

无论你将Synchronized加在方法前还是加在一个变量前,其锁定的都是一个 类对象。 每一个对象都只有一个锁与之相关联。

下例中分情况的列举各种情况下的同步效果

1. Synchronized 加在方法上, (同步方法,锁定类实例)

Java代码

publicclassDemo1 {

publicsynchronizedvoidm1(){

//...............

}

publicvoidm2(){

//............

synchronized(this){

//.........

}

//........

}

}

这两种写法的效果是一样的,锁定的都是类实例对象。如果有一个 类实例对象: demo = new Demo1(),另外有两个线程: thread1,thread2,都调用了demo 对象,那么,在同一时间,如果 thread1调用了demo.m1(),则thread2在该时间内不能访问demo.m1() 和 demo.m2(); 因为thread1把demo这个对象的锁使用了,所以无法分给其它线程使用

但是,如果thread1调用 demo1.m1(), thread2调用 demo2.m1(), 则可以同时进行,因为它们调用的是不同的Demo1类对象实例。

2. Synchronized 加在变量上, (同步块,锁定类实例)

Java代码

publicclassDemo2 {

Object a =newObject();

Object b =newObject();

publicvoidm1(){

//............

synchronized(a){

//.........

}

//........

}

publicvoidm2(){

//............

synchronized(b){

//.........

}

//........

}

}

这种情况下,是实现代码块锁定,锁定的对象是 变量 a 或 b; (注意,a 、b 都是非static 的)如果有一个 类实例对象: demo = new Demo2(),另外有两个线程: thread1,thread2,都调用了demo 对象,那么,在同一时间,如果 thread1调用了demo.m1(),则thread2在该时间内可以访问demo.m2();但不能访问 demo.m1() 的同步块, 因为a被 thread1锁定了。

3. Synchronized 锁定的是 类变量 ,即static 变量(可能是属性,可能是方法)(锁定类对象)

Java代码

publicclassDemo3 {

staticObject o =newObject();

publicstaticsynchronizedvoidm1() {

//....

}

publicstaticvoidm2() {

//...

synchronized(Demo3.class) {

//.....

}

//.....

}

publicstaticvoidm3() {

//..........

try{

synchronized(Class.forName("Demo3")) {

//............

}

}catch(ClassNotFoundException ex) {

}

//.............

}

publicstaticvoidm4() {

//............

synchronized(o){

//........

}

//..........

}

}

以上4个方法中实现的效果都是一样的,其锁定的对象都是类Demo3,而不是类实例对象 ,即在多线程中,其共享的资源是属于类的,而不是属于类对象的。在这种情况下,如果thread1 访问了这4个方法中的任何一个, 在同一时间内其它的线程都不能访问 这4个方法。

4. 类的方法中访问了多线程共同的资源, 且该资源是可变的,这种情况下也是需要进行同步的

Java代码

publicclassDemo4 {

staticString path ="file path";

publicvoidreadConfiFile() {

synchronized(path) {

// 读取该path指定的文件。

}

}

publicvoidwriteConfiFile() {

synchronized(path) {

//写信息到该path指定的文件。

}

}

}

这种情况下,必须锁定为 类变量,而不能进行锁定类实例对象,因为这是变象的一种类资源共享,而不是类实例对象资源共享。

线程,成也其,败也其,用好了可以提升性能,用不好则会使系统后患无穷。

PS: 进行线程同步需要很大的系统开销, 所以,在使用时,如果不是必须的,则尽量不使用同步功能。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,518评论 1 15
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,835评论 18 399
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 3,017评论 1 18
  • 写在前面的话: 这篇博客是我从这里“转载”的,为什么转载两个字加“”呢?因为这绝不是简单的复制粘贴,我花了五六个小...
    SmartSean阅读 4,821评论 12 45
  • 我们在修行的过程中,总是苦于没有老师的指导。实际上,我们每天咏念的《心经》,就是我们修行、生活最好的指导,最亲切的...
    旺财杨子阅读 2,270评论 2 50