RELATEED CONSULTING
相关咨询
选择下列产品马上在线沟通
服务时间:9:30-18:00
你可能遇到了下面的问题
关闭右侧工具栏
破除java神话之四:同步代码等同于断面(critic
  • 作者:xiaoxiao
  • 发表时间:2020-12-23 11:01
  • 来源:未知

同步经常作为断面被引用。断面是指一次只能有一个线程执行它。多个线程同时执行同步代码是有可能的。这个误解是因为很多程序员认为同步关键字锁住了它所包围的代码。但是实际情况不是这样的。同步加锁的是对象,而不是代码。因此,如果你的类中有一个同步方法,这个方法可以被两个不同的线程同时执行,只要每个线程自己创建一个的该类的实例即可。参考下面的代码:class Foo extends Thread {   private int val;   public Foo(int v)   {    val = v;   }   public synchronized void printVal(int v)   {    while(true)     System.out.println(v);   }   public void run()   {    printVal(val);   }  }class SyncTest {  public static void main(String args[])  {   Foo f1 = new Foo(1);   f1.start();   Foo f2 = new Foo(3);  f2.start();  } } 运行SyncTest产生的输出是1和3交叉的。如果printVal是断面,你看到的输出只能是1或者只能是3而不能是两者同时出现。程序运行的结果证明两个线程都在并发的执行printVal方法,即使该方法是同步的并且由于是一个无限循环而没有终止。要实现真正的断面,你必须同步一个全局对象或者对类进行同步。下面的代码给出了一个这样的范例。class Foo extends Thread {  private int val;  public Foo(int v)  {   val = v;  }  public void printVal(int v)  {   synchronized(Foo.class) {    while(true)    System.out.println(v);   }  }  public void run()  {   printVal(val);  } } 上面的类不再对个别的类实例同步而是对类进行同步。对于类Foo而言,它只有唯一的类定义,两个线程在相同的锁上同步,因此只有一个线程可以执行printVal方法。这个代码也可以通过对公共对象加锁。例如给Foo添加一个静态成员。两个方法都可以同步这个对象而达到线程安全。译者注:下面笔者给出一个参考实现,给出同步公共对象的两种通常方法:1、class Foo extends Thread {  private int val; private static Object lock=new Object(); public Foo(int v)  {   val = v;  }  public void printVal(int v)  {   synchronized(lock) {    while(true)    System.out.println(v);   }  }  public void run()  {   printVal(val);  } }上面的这个例子比原文给出的例子要好一些,因为原文中的加锁是针对类定义的,一个类只能有一个类定义,而同步的一般原理是应该尽量减小同步的粒度以到达更好的性能。笔者给出的范例的同步粒度比原文的要小。2、