synchronized作用:
Synchronized关键字的作用一个词概括就是:线程同步。它可以用来修改对象中的方法,将对象加锁。相当于不管哪一个线程A每次运行到这个方法时,都要检查有没有其它正在用这个方法的线程B(或者C D等),有的话要等正在使用这个方法的线程B(或者C D)运行完这个方法后再运行此线程A,没有的话,直接运行。
synchronized语法:
1.synchronized方法
如:public synchronized void accessVal(int newVal);
synchronized方法控制对类成员变量的访问:每个类实例对应一把锁,每个synchronized方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。这种机制确保了同一时刻对于每一个类实例,其所有声明为synchronized的成员函数中至多只有一个处于可执行状态(因为至多只有一个能够获得该类实例对应的锁),从而有效避免了类成员变量的访问冲突(只要所有可能访问类成员变量的方法均被声明为synchronized)。在java中,不光是类实例,每一个类也对应一把锁,这样我们也可将类的静态成员函数声明为synchronized,以控制其对类的静态成员变量的访问。
2.synchronized块
synchronized方法是对整个方法进行加锁。若将一个大的方法声明为synchronized将会大大影响效率,典型地,若将线程类的方法run()声明为synchronized,由于在线程的整个生命期内它一直在运行,因此将导致它对本类任何synchronized方法的调用都永远不会成功。当然我们可以通过将访问类成员变量的代码放到专门的方法中,将其声明为synchronized,并在主方法中调用来解决这一问题,但是java为我们提供了更好地解决办法,那就是synchronized块。
如:synchronized(syncObject){
//允许访问控制的代码
}
synchronized块是这样一个代码块,其中的代码必须获得对象syncObject(如前所述,可以是类实例或类)的锁方能执行,具体机制同前所述。由于可以针对任意代码块,且可任意指定上锁对象,故灵活性较高。
sysnchronized(this):
1.当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。
1 public class Thread1 implements Runnable { 2 3 public void run(){ 4 synchronized(this){ 5 for(int i=0;i<5;i++){ 6 System.out.println(Thread.currentThread().getName() + " synchronized loop " + i); 7 } 8 } 9 } 10 11 public static void main(String[] args) { 12 Thread1 t1 = new Thread1(); 13 Thread ta = new Thread(t1,"A"); 14 Thread tb = new Thread(t1,"B"); 15 ta.start(); 16 tb.start(); 17 } 18 }
2.然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码。
1 public class Thread2 { 2 3 public void m4t1(){ 4 synchronized(this){ 5 int i = 5; 6 while(i-->0){ 7 System.out.println(Thread.currentThread().getName() + " : " + i); 8 try{ 9 Thread.sleep(500); 10 }catch(InterruptedException ie){ 11 ie.printStackTrace(); 12 } 13 } 14 } 15 } 16 17 public void m4t2(){ 18 int i = 5; 19 while(i-->0){ 20 System.out.println(Thread.currentThread().getName() + " : " + i); 21 try{ 22 Thread.sleep(500); 23 }catch(InterruptedException ie){ 24 ie.printStackTrace(); 25 } 26 } 27 } 28 29 public static void main(String[] args) { 30 31 final Thread2 myt2 = new Thread2(); 32 Thread t1 = new Thread( 33 new Runnable(){ 34 public void run(){ 35 myt2.m4t1(); 36 } 37 },"t1" 38 ); 39 Thread t2 = new Thread( 40 new Runnable(){ 41 public void run(){ 42 myt2.m4t2(); 43 } 44 },"t2" 45 ); 46 t1.start(); 47 t2.start(); 48 } 49 50 }
3.尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其它线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。
1 public class Thread2 { 2 3 public void m4t1(){ 4 synchronized(this){ 5 int i = 5; 6 while(i-->0){ 7 System.out.println(Thread.currentThread().getName() + " : " + i); 8 try{ 9 Thread.sleep(500); 10 }catch(InterruptedException ie){ 11 ie.printStackTrace(); 12 } 13 } 14 } 15 } 16 17 public void m4t2(){ 18 synchronized(this){ 19 int i = 5; 20 while(i-->0){ 21 System.out.println(Thread.currentThread().getName() + " : " + i); 22 try{ 23 Thread.sleep(500); 24 }catch(InterruptedException ie){ 25 ie.printStackTrace(); 26 } 27 } 28 } 29 } 30 31 public static void main(String[] args) { 32 33 final Thread2 myt2 = new Thread2(); 34 Thread t1 = new Thread( 35 new Runnable(){ 36 public void run(){ 37 myt2.m4t1(); 38 } 39 },"t1" 40 ); 41 Thread t2 = new Thread( 42 new Runnable(){ 43 public void run(){ 44 myt2.m4t2(); 45 } 46 },"t2" 47 ); 48 t1.start(); 49 t2.start(); 50 } 51 52 }