简体   繁体   English

如何在 Java 中运行类的不同实例的线程之间同步静态变量?

[英]How to synchronize a static variable among threads running different instances of a class in Java?

I know that using the synchronize keyword before a method brings synchronization to that object.我知道在方法为该对象带来同步之前使用synchronize关键字。 That is, 2 threads running the same instance of the object will be synchronized.也就是说,运行同一对象实例的 2 个线程将被同步。

However, since the synchronization is at the object level, 2 threads running different instances of the object will not be synchronized.但是,由于同步是在对象级别,因此运行对象的不同实例的 2 个线程将不会同步。 If we have a static variable in a Java class that is called by the method, we would like it to be synchronized across instances of the class.如果我们在方法调用的 Java 类中有一个静态变量,我们希望它在类的实例之间同步。 The two instances are running in 2 different threads.这两个实例在 2 个不同的线程中运行。

Can we achieve synchronization in the following way?我们可以通过以下方式实现同​​步吗?

public class Test  
{  
   private static int count = 0;  
   private static final Object lock= new Object();    
   public synchronized void foo() 
  {  
      synchronized(lock)
     {  
         count++;  
     }  
  }  
}

Is it true that since we have defined an object lock that is static and we are using the keyword synchronized for that lock, the static variable count is now synchronized across instances of class Test ?是不是因为我们已经定义了一个静态的对象lock ,并且我们正在为该锁使用关键字synchronized ,那么静态变量count现在在类Test实例之间同步?

There are several ways to synchronize access to a static variable.有几种方法可以同步对静态变量的访问。

  1. Use a synchronized static method.使用同步静态方法。 This synchronizes on the class object.这在类对象上同步。

     public class Test { private static int count = 0; public static synchronized void incrementCount() { count++; } }
  2. Explicitly synchronize on the class object.在类对象上显式同步。

     public class Test { private static int count = 0; public void incrementCount() { synchronized (Test.class) { count++; } } }
  3. Synchronize on some other static object.同步其他一些静态对象。

     public class Test { private static int count = 0; private static final Object countLock = new Object(); public void incrementCount() { synchronized (countLock) { count++; } } }

Method 3 is the best in many cases because the lock object is not exposed outside of your class.方法 3 在许多情况下是最好的,因为锁对象不会暴露在您的类之外。

If you're simply sharing a counter, consider using an AtomicInteger or another suitable class from the java.util.concurrent.atomic package:如果您只是共享一个计数器,请考虑使用AtomicInteger或 java.util.concurrent.atomic 包中的其他合适的类:

public class Test {

    private final static AtomicInteger count = new AtomicInteger(0); 

    public void foo() {  
        count.incrementAndGet();
    }  
}

Yes it is true.是的,它是真的。

If you create two instance of your class如果您创建类的两个实例

Test t1 = new Test();
Test t2 = new Test();

Then t1.foo and t2.foo both synchronize on the same static object and hence block each other.然后 t1.foo 和 t2.foo 都在同一个静态对象上同步,因此相互阻塞。

We can also use ReentrantLock to achieve the synchronization for static variables.我们也可以使用ReentrantLock来实现静态变量的同步。

public class Test {

    private static int count = 0;
    private static final ReentrantLock reentrantLock = new ReentrantLock(); 
    public void foo() {  
        reentrantLock.lock();
        count = count + 1;
        reentrantLock.unlock();
    }  
}

You can synchronize your code over the class.您可以在类上同步您的代码。 That would be simplest.那将是最简单的。

   public class Test  
    {  
       private static int count = 0;  
       private static final Object lock= new Object();    
       public synchronized void foo() 
      {  
          synchronized(Test.class)
         {  
             count++;  
         }  
      }  
    }

Hope you find this answer useful.希望你觉得这个答案有用。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM