學習java synchronized的總結

1. synchronized 可以作爲一個method的修飾符
 public synchronized void aMethod() {
        //some code
    }
相當於:
    public void aMethod() {
        synchronized(this) {
           //some code
        }
    }
說明:
(1) 它是對類中的同一對象負責
(2)在同一個類對象實例內,synchronized aMethod(){}可以防止多個線程同時訪問這個對象的synchronized方法
舉例:
public class Thread1 extends Thread{
 
 public void run(){
  /*synchronized (this) {
   for (int i = 0; i < 5; i++) {
    try {
     Thread.sleep(1000);
    } catch (InterruptedException e) {
     // TODO Auto-generated catch block
     e.printStackTrace();
    }
    System.out.println(Thread.currentThread().getName() + " " + i);
   }
  }*/
  myRun();
 }
 
 //與synchronized(this){}效果是一致的
 public synchronized void myRun(){
  for (int i = 0; i < 5; i++) {
   try {
    Thread.sleep(1000);
   } catch (InterruptedException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
   }
   System.out.println(Thread.currentThread().getName() + " " + i);
  }
 }

 public static void main(String[] args) {
  Thread1 t = new Thread1();
  Thread myA = new Thread(t,"A");
  Thread myB = new Thread(t,"B");
  
  myA.start();
  myB.start();
  
 }
運行結果:
A 0
A 1
A 2
A 3
A 4
B 0
B 1
B 2
B 3
B 4

(3) 如果是一個類中的不同對象,多個線程可以同時訪問synchronized方法:
將main()方法修改:
  Thread1 t1 = new Thread1();
  Thread1 t2 = new Thread1();
  
  t1.setName("t1");
  t2.setName("t2");
  
  t1.start();
  t2.start();
運行結果:
t1 0
t2 0
t1 1
t2 1
t1 2
t2 2
t1 3
t2 3
t1 4
t2 4

(4) 如果是類的同一個對象存在多個synchronized方法,只要一個線程訪問了其中的一個synchronized方法,其它線程不能同時訪問這個對象 中任何一個synchronized方法,但可以訪問非synchronized方法。
public class Thread2 extends Thread {

 /**
  * (1) 當一個線程訪問object的一個synchronized(this)同步代碼塊時,
  * 另一個線程仍然可以訪問該object中的非synchronized(this)同步代碼塊
  *
  * (2)當一個線程訪問object的一個synchronized(this)同步代碼塊時,
  * 其他線程對object中所有其它synchronized(this)同步代碼塊的訪問將被阻塞
  * @param args
  */
 
 public synchronized void method1(){
  for (int i = 0; i < 5; i++) {
   try {
    Thread.sleep(1000);
   } catch (InterruptedException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
   }
   System.out.println(Thread.currentThread().getName() + " " + i);
  }
 }
 
 public synchronized void method2(){
  for (int i = 0; i < 5; i++) {
   try {
    Thread.sleep(1000);
   } catch (InterruptedException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
   }
   System.out.println(Thread.currentThread().getName() + " " + i);
  }
 }
 
 public static void main(String[] args) {
  final Thread2 t2 = new Thread2();
  Thread myThreadA = new Thread(
    new Runnable(){
     public void run(){
      t2.method1();
     }
    },"myThreadA");
  
  Thread myThreadB = new Thread(
    new Runnable(){
     public void run(){
      t2.method2();
     }
    },"myThreadB");
  
  myThreadA.start();
  myThreadB.start();
 }

}
運行結果:
myThreadA 0
myThreadA 1
myThreadA 2
myThreadA 3
myThreadA 4
myThreadB 0
myThreadB 1
myThreadB 2
myThreadB 3
myThreadB 4

2. synchronized修飾一個 static的method,
如:
    public static synchronized void aMethod() {
        //some code
    }
相當於:
    public static synchronized void aMethod() {
        synchronized(XX.class) {
           //some code
        }
    }
說明:
synchronized static aStaticMethod{}防止多個線程同時訪問這個類中的synchronized static 方法。它可以對類的所有對象實例起作用。
public class ThreadTestE extends Thread {

 public void run(){
  myRun();
 }
 
 public static synchronized void myRun(){
  for (int i = 0; i < 5; i++) {
   try {
    Thread.sleep(1000);
   } catch (InterruptedException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
   }
   System.out.println(Thread.currentThread().getName() + " " + i);
  }
 }
 
 public static void main(String[] args) {
  ThreadTestE e1 = new ThreadTestE();
  ThreadTestE e2 = new ThreadTestE();
  ThreadTestE e3 = new ThreadTestE();
  
  e1.start();
  e2.start();
  e3.start();
 
 }

}
運行結果:
Thread-0 0
Thread-0 1
Thread-0 2
Thread-0 3
Thread-0 4
Thread-1 0
Thread-1 1
Thread-1 2
Thread-1 3
Thread-1 4
Thread-2 0
Thread-2 1
Thread-2 2
Thread-2 3
Thread-2 4

總結:
無論synchronized關鍵字加在方法上還是對象上,它取得的鎖都是對象。
synchronized(object){
}
線程獲得了object對象鎖,執行裏面的代碼,其它線程阻塞,直至其它線程也獲得了該對象鎖才能執行裏面的代碼;

補充說明:
synchronized關鍵字是不能繼承的,也就是說,基類的方法synchronized f(){} 在繼承類中並不自動是synchronized f(){},而是變成了f() {}。繼承類需要你顯式的指定它的某個方法爲synchronized方法

參考文章:
(1)java synchronized 用法 http://www.360doc.com/showWeb/0/0/493058.aspx
(2)java synchronized的幾個規則  http://blog.donews.com/dyhcn/archive/2005/07/14/465420.aspx

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章