随笔 - 3, 文章 - 152, 评论 - 17, 引用 - 0
数据加载中……

对Java同步一些理解

对Java同步一些理解
kingfish 2005
------------------------------------------------------------
基本概念:
  每个Object都会有1个锁.
  同步就是串行使用一些资源.

(说明:以下有些例子为了突出重点,省略了不必要的代码.特别是省掉了一些成员变量,就是需要同步的对象.)

1. 多线程中对共享、可变的数据进行同步.
   对于函数中的局部变量没必要进行同步.
   对于不可变数据,也没必要进行同步.

   多线程中访问共享可变数据才有必要.


2. 单个线程中可以使用synchronized,而且可以嵌套,但无意义.
   class Test {
     public static void main(String[] args) {
         Test t = new Test();
         synchronized(t) {
           synchronized(t) {
              System.out.println("ok!");
           }
         }
     }
   }
  
3. 对象实例的锁
   class Test{
     public synchronized void f1(){
         //do something here
     }

     public void f2(){
        synchronized(this){
           //do something here
         }
     }
   }
 
   上面的f1()和f2()效果一致, synchronized取得的锁都是Test某个实列(this)的锁.
   比如: Test t = new Test();
         线程A调用t.f2()时, 线程B无法进入t.f1(),直到t.f2()结束.

   作用: 多线程中访问Test的同一个实例的同步方法时会进行同步.

4. class的锁
   class Test{
       final static Object o= new Object();

       public static synchronized void f1(){
          //do something here
       }
 
       public static void f2(){
         synchronized(Test.class){
           //do something here
         }
       }
  
      public static void f3(){
        try {
          synchronized (Class.forName("Test")) {
          //do something here
          }
        }
        catch (ClassNotFoundException ex) {
        }
      }
     
       public static void g(){
          synchronized(o){
            //do something here
          }
       }
   }

   上面f1(),f2(),f3(),g()效果一致
   f1(),f2(),f3()中synchronized取得的锁都是Test.class的锁.
  
   g()是自己产生一个对象o,利用o的锁做同步
   作用: 多线程中访问此类或此类任一个实例的同步方法时都会同步. singleton模式lazily initializing属于此类.

5. static method
   class Test{
      private static int v = 0;

      public static void f1(){
        //do something, 但函数中没用用到v
      }

      public synchronized static void f2(){
        //do something, 函数中对v进行了读/写.
      }  
   }

   多线程中使用Test的某个实列时,
    (1) f1()是线程安全的,不需要同步
    (2) f2()这个静态方法中使用了函数外静态变量,所以需要同步.

6.  对线程的run()进行同步没有意义,如 public synchronized void run()
    class Test extends Thread{
      public synchronized void run(){
        while(true){
           //do something
        }
      }
     
      public synchronized void f(){
         //...
      }
    }
    这种例子会有一个问题, 执行run()时(内部在循环), 外部无法执行f()


    class Test extends Thread{
       public synchronized void run(){
         //do something
       }
    }
    这种例子同步基本没用, 因为run()通常靠 new Test().start()来执行的.
    因为Test实例不同,锁也不同.

posted on 2005-02-27 17:37 阅读(225) 评论(0)  编辑  收藏 所属分类: J2se


只有注册用户登录后才能发表评论。


网站导航: