线程同步的2种方法

来源:互联网 发布:香港大学网络教育 编辑:程序博客网 时间:2024/05/22 21:54

线程同步的2种方法

==============

同步的实现方式有两种,同步方法和同步块,这两种方式都要用到synchronized关键字。


(1)同步方法
给一个方法增加synchronized修饰符之后就可以使它成为同步方法,这个方法可以是静态方法和非静态方法,
但是不能是抽象类的抽象方法,也不能是接口中的接口方法。
下面代码是一个同步方法的示例:


public synchronized void aMethod() {     // do something } public static synchronized void anotherMethod() {     // do something } 



        线程在执行同步方法时是具有排它性的。当任意一个线程进入到一个对象的任意一个同步方法时,这个对象的所有同步方法都被锁定了,在此期间,其他任何线程都不能访问这个对象的任意一个同步方法,直到这个线程执行完它所调用的同步方法并从中退出,从而导致它释放了该对象的同步锁之后。在一个对象被某个线程锁定之后,其他线程是可以访问这个对象的所有非同步方法的。


(2)同步块
        同步块是通过锁定一个指定的对象,来对同步块中包含的代码进行同步;而同步方法是对这个方法块里的代码进行同步,而这种情况下锁定的对象就是同步方法所属的主体对象自身。如果这个方法是静态同步方法呢?那么线程锁定的就不是这个类的对象了,也不是这个类自身,而是这个类对应的java.lang.Class类型的对象。同步方法和同步块之间的相互制约只限于同一个对象之间,所以静态同步方法只受它所属类的其它静态同步方法的制约,而跟这个类的实例(对象)没有关系。
       如果一个对象既有同步方法,又有同步块,那么当其中任意一个同步方法或者同步块被某个线程执行时,这个对象就被锁定了,其他线程无法在此时访问这个对象的同步方法,也不能执行同步块。
        在可能的情况下,应该把保护范围缩到最小,可以用示例4 的形式,this 代表“这个对象”。没有必要把整个run()保护起来,run()中的代码只有一个for循环,所以只要保护for 循环就可以了。


示例:

public class ThreadTest implements Runnable{public void run(){   for(int k=0;k<5;k++){       System.out.println(Thread.currentThread().getName()+ " : for loop : " + k);   }   synchronized(this){      for(int k=0;k<5;k++) {         System.out.println(Thread.currentThread().getName()+ " : synchronized for loop : " + k);   }}}public static void main(String[] args){  Runnable r = new ThreadTest();Thread t1 = new Thread(r,"t1_name");Thread t2 = new Thread(r,"t2_name");t1.start();t2.start();} }

同步块的另一个思路:

package com.java.thread;public class ThreadTest2 extends Thread {private int threadNo;private String lock;public ThreadTest2(int threadNo, String lock) {this.threadNo = threadNo;this.lock = lock;}public static void main(String[] args) throws Exception {String lock = new String("lock");for (int i = 1; i < 10; i++) {new ThreadTest2(i, lock).start();Thread.sleep(1);}}public void run() {synchronized (lock) {for (int i = 1; i < 4; i++) {System.out.println("No." + threadNo + ":" + i);}}}}




0 0