java同步机制:synchronized
2012-03-17 16:55
357 查看
原文地址:/article/1425315.html
synchronized 是java语言关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。synchronized 关键字,它包括两种用法:synchronized 方法和 synchronized 块。
本文直接以代码的形式来展示 synchronized 关键字的使用:
【1】synchronized Demo1:
运行结果如下:
【2】synchronized Demo2:
运行结果如下:
【3】synchronized Demo3:
运行结果如下:
总结:
1. synchronized 方法控制对类成员变量的访问:每个类实例对应一把锁,每个 synchronized 方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。这种机制确保了同一时刻对于每一个类实例,其所有声明为 synchronized 的成员函数中至多只有一个处于可执行状态(因为至多只有一个能够获得该类实例对应的锁),从而有效避免了类成员变量的访问冲突(只要所有可能访问类成员变量的方法均被声明为
synchronized)。
2. synchronized 块是这样一个代码块,其中的代码必须获得对象 syncObject (如前所述,可以是类实例或类)的锁方能执行。由于可以针对任意代码块,且可任意指定上锁的对象,故灵活性较高。
对synchronized(this)的一些理解
一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。
二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。
三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。
四、当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。
synchronized 是java语言关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。synchronized 关键字,它包括两种用法:synchronized 方法和 synchronized 块。
本文直接以代码的形式来展示 synchronized 关键字的使用:
【1】synchronized Demo1:
package com.andyidea.demo; /** * 当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时, * 一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码 * 块以后才能执行该代码块。 * @author Andy.Chen * */ public class Thread01 implements Runnable { @Override public void run() { synchronized (this) { for(int i=0;i<3;i++){ System.out.println(Thread.currentThread().getName()+" synchronized loop "+i); } } } public static void main(String[] args) { Thread01 t01 = new Thread01(); System.out.println("Welcome to Andy.Chen Blog! \n" +"synchronized 关键字使用 \n" +"--------------------------"); Thread ta = new Thread(t01,"A"); Thread tb = new Thread(t01,"B"); ta.start(); tb.start(); } }
运行结果如下:
Welcome to Andy.Chen Blog! synchronized 关键字使用 -------------------------- B synchronized loop 0 B synchronized loop 1 B synchronized loop 2 A synchronized loop 0 A synchronized loop 1 A synchronized loop 2
【2】synchronized Demo2:
package com.andyidea.demo; /** * 当一个线程访问object的一个synchronized(this)同步代码块时, * 另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。 * @author Andy.Chen * */ public class Thread02 { public void method01(){ synchronized (this) { int i=0; while(i++ < 3){ System.out.println(Thread.currentThread().getName() +":"+ i); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } } } public void method02(){ //第1种方式:当一个线程访问object的一个synchronized(this)同步代码块时, //另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。 // int j=0; // while(j++ < 3){ // System.out.println(Thread.currentThread().getName() +":"+ j); // try { // Thread.sleep(1000); // } catch (InterruptedException e) { // e.printStackTrace(); // } // } //第2种方式:当一个线程访问object的一个synchronized(this)同步代码块时, //其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。 synchronized (this) { int j=0; while(j++ < 3){ System.out.println(Thread.currentThread().getName() +":"+ j); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } } } /** * 当一个线程访问object的一个synchronized(this)同步代码块时, * 它就获得了这个object的对象锁。 * 结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。 */ public synchronized void method3(){ int k=0; while(k++ < 3){ System.out.println(Thread.currentThread().getName() +":"+ k); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } } public static void main(String[] args) { final Thread02 t02 = new Thread02(); System.out.println("Welcome to Andy.Chen Blog! \n" +"synchronized 关键字使用 \n" +"--------------------------"); Thread t02A = new Thread(new Runnable() { @Override public void run() { t02.method01(); } },"A"); Thread t02B = new Thread(new Runnable() { @Override public void run() { t02.method02(); } },"B"); Thread t02C = new Thread(new Runnable() { @Override public void run() { t02.method3(); } },"C"); t02A.start(); t02B.start(); t02C.start(); } }
运行结果如下:
Welcome to Andy.Chen Blog! synchronized 关键字使用 -------------------------- B:1 B:2 B:3 C:1 C:2 C:3 A:1 A:2 A:3
【3】synchronized Demo3:
package com.andyidea.demo; /** * synchronized对象锁 * @author Andy.Chen * */ public class Thread03 { class InnerObject{ /** * 内部类方法1 */ private void innerMethod01(){ int i=0; while(i++ < 3){ System.out.println(Thread.currentThread().getName() +":"+ i); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } } /** * 内部类方法2 */ private void innerMethod02(){ int j=0; while(j++ < 3){ System.out.println(Thread.currentThread().getName() +":"+ j); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } } } /** * 外部类方法1 * @param innerObj */ private void outerMethod01(InnerObject innerObj){ synchronized (innerObj) { innerObj.innerMethod01(); } } /** * 外部类方法2 * @param innerObj */ private void outerMethod02(InnerObject innerObj){ innerObj.innerMethod02(); } public static void main(String[] args) { final Thread03 t03 = new Thread03(); final InnerObject innerObj = t03.new InnerObject(); System.out.println("Welcome to Andy.Chen Blog! \n" +"synchronized 关键字使用 \n" +"--------------------------"); Thread t03A = new Thread(new Runnable() { @Override public void run() { t03.outerMethod01(innerObj); } },"A"); Thread t03B = new Thread(new Runnable() { @Override public void run() { t03.outerMethod02(innerObj); } },"B"); t03A.start(); t03B.start(); } }
运行结果如下:
Welcome to Andy.Chen Blog! synchronized 关键字使用 -------------------------- A:1 B:1 B:2 A:2 B:3 A:3
总结:
1. synchronized 方法控制对类成员变量的访问:每个类实例对应一把锁,每个 synchronized 方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。这种机制确保了同一时刻对于每一个类实例,其所有声明为 synchronized 的成员函数中至多只有一个处于可执行状态(因为至多只有一个能够获得该类实例对应的锁),从而有效避免了类成员变量的访问冲突(只要所有可能访问类成员变量的方法均被声明为
synchronized)。
2. synchronized 块是这样一个代码块,其中的代码必须获得对象 syncObject (如前所述,可以是类实例或类)的锁方能执行。由于可以针对任意代码块,且可任意指定上锁的对象,故灵活性较高。
对synchronized(this)的一些理解
一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。
二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。
三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。
四、当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。
相关文章推荐
- java同步机制 synchronized
- Java同步机制浅谈――synchronized对代码作何影响?
- Java的多线程机制系列:(三)synchronized的同步原理
- Java线程同步机制synchronized关键字的理解
- java同步机制 synchronized
- 转载:Java同步机制浅谈――synchronized对代码作何影响?
- java同步机制 synchronized (转)
- java同步机制synchronized
- 关于Java同步机制中synchronized同步方法
- Java的synchronized关键字:同步机制总结
- Java同步机制总结--synchronized
- 秋招准备-Java-并发编程-同步机制与synchronized(二)
- java两种同步机制的实现 synchronized和reentrantlock
- java同步机制:synchronized
- Java同步机制――synchronized对代码作何影响
- 【Android】java同步机制:synchronized
- 浅谈Java多线程同步机制之同步块(方法)——synchronized
- Java的synchronized关键字:同步机制总结
- java多线程、并发系列之 (synchronized)同步与加锁机制
- Java的synchronized关键字:同步机制总结