学习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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值