在getter / setter方法和一类的单例模式的Java细粒度同步(Java fine-gra

2019-09-21 10:02发布

我试图用synchronization的Java指令来实现一类细粒度同步,即同步的代码,我可以量最少的。我会注释代码内联,来解释我做什么和代码后,我”会问你如何改进的代码:

public class MyClass {
    private static volatile MyClass singletonInstance = null;

    private HashMap<String, Integer> mHashMap = null;
    private String mStringA = null;
    private String mStringB = null;


     // Use double check technique to use synchronization only 
     // at the first getInstance() invocation
    public static MyClass getInstance() {
        if (singletonInstance == null) {
            synchronized (MyClass.class) {
                if (singletonInstance == null)
                    singletonInstance = new MyClass();
                    // Initialize class member variables
                    singletonInstance.mHashMap = new HashMap<String,Integer>();
                    singletonInstance.mStringA = new String();
                    singletonInstance.mStringB = new String();
            }

        }
        return singletonInstance;
    }

    // The following two methods manipulate the HashMap mHashMap
    // in a secure way since they lock the mHashMap instance which
    // is always the same and is unique
    public Integer getIntegerFromHashmap(String key) {
        synchronized (mHashMap) {
            return mHashMap.get(key);
        }
    }

    public void setIntegerIntoHashmap(String key, Integer value) {
        synchronized (mHashMap) {
            mHashMap.put(key, value);
        }
    }

    // With the two String members mStringA and mStringB the problem is 
    // that the instance of String pointed by the member is varied by the 
    // setter methods, so we can not lock in a fine grained way and we
    // must lock on the singletonInstance.
    public String getStringA() {
        synchronized (singletonInstance) {
            return mStringA;
        }
    }

    public String getStringB() {
        synchronized (singletonInstance) {
            return mStringB;
        }
    }

    public void setStringA(String newString) {
        synchronized (singletonInstance) {
            mStringA = newString;
        }
    }

    public void setStringB(String newString) {
        synchronized (singletonInstance) {
            mStringB = newString;
        }
    }
}

我不喜欢两个的getter和setter方法String成员变量是锁定在singletonInstance可以让一个线程试图访问mStringB等到被操纵线程mStringA释放其锁。 你会在这种情况下怎么办? 请问你创建了两个成员变量private final Integer mStringALock = new Integer(0)private final Integer mStringBLock = new Integer(0)MyClass和的getter和setter方法synchronized块使用它们mStringAmStringB分别?

如果您有关于如何改进上面的代码,为的细粒度同步变化提出一些想法String成员变量,欢迎您:)

Answer 1:

通常,简单的解决方案更容易实现。 我也将使用并行库在2004年加入。

这并不需要显式的锁,每个容器是线程安全的。

您可以使用的AtomicReference但在这种情况下,它不会给你任何挥发性不给你了。 (由于kdgregory指出的),你可能会在更复杂的情况下使用的AtomicReference。

public enum MyClass {
    INSTANCE;

    private final Map<String, Integer> mHashMap = new ConcurrentHashMap<String, Integer>();
    private volatile String mStringA = null;
    private volatile String mStringB = null;


    // The following two methods manipulate the HashMap mHashMap
    // in a secure way
    public Integer getIntegerFromHashmap(String key) {
        return mHashMap.get(key);
    }

    public void setIntegerIntoHashmap(String key, Integer value) {
        mHashMap.put(key, value);
    }

    public String getStringA() {
        return mStringA;
    }

    public String getStringB() {
        return mStringB;
    }

    public void setStringA(String newString) {
        mStringA = newString;
    }

    public void setStringB(String newString) {
        mStringB = newString;
    }
}


Answer 2:

是的,你如果希望线程能够在两种方法同时调用需要两个单独的锁。 我会说没有什么改善这个问题。

然而,我注意到,您getInstance()方法试图最小化块的大小同步,但你居然不做到这一点,即你检查singletonInstance == null synchronized块里面了。 所以,我认为这将是更好的资格整个方法与synchronized

它缩短了方法的代码,它使得它更有点顺理成章了。



Answer 3:

从哪里开始?

OK,双重检查锁定:它坏了(还),不使用它。 如果你觉得你必须使用一个单身(真的,他们一般是一个坏主意,使用依赖注入代替),然后同步getter方法,并快速返回。 除非你有内核的真正数量庞大(如,千),并不断调用getter方法在这种情况下,争议同步的可能性非常低。

更换HashMap一个ConcurrentHashMap 。 Doug Lea的是比你或我并行编码更好。

标记字符串变量易失性和不同步它们。



文章来源: Java fine-grained synchronization in getter/setter methods and singleton pattern of a class