如何正确使用VAVR集合是线程安全的?

时间:2018-02-21 14:20:50

标签: java thread-safety vavr

VAVR集合是“不可变的”。

所以,如果我有静态变量,例如,持有所有WebSocket会话,我将如何使用VAVR以使该集合是线程安全的?

例如:

@ServerEndpoint("/actions")
public class DeviceWebSocketServer {

    private static Set<Session> sessions = //???; // how should I initialize this?

    @OnOpen
    public void open(Session session) {
        sessions = sessions.add(session); // is this OK???
    }

    @OnClose
    public void close(Session session) {
        sessions = sessions.remove(session); // is this OK??
    }
}    

2 个答案:

答案 0 :(得分:7)

您可以将不可变的vavr集合包装在原子可更新的AtomicReference中,并使用其中一种更新方法以原子方式更新对不可变集合的引用。

@ServerEndpoint("/actions")
public class DeviceWebSocketServer {

    private static AtomicReference<Set<Session>> sessionsRef = 
            new AtomicReference<>(HashSet.empty());

    @OnOpen
    public void open(Session session) {
        sessionsRef.updateAndGet(sessions -> sessions.add(session));
    }

    @OnClose
    public void close(Session session) {
        sessionsRef.updateAndGet(sessions -> sessions.remove(session));
    }

}

如果要在其他场景中使用它们,请确保阅读AtomicReference的javadoc,因为对更新函数有一些要求需要遵守才能获得正确的行为。

答案 1 :(得分:-1)

对于此用例,您还可以考虑使用并发映射:

@ServerEndpoint("/actions")
public class DeviceWebSocketServer {

    private static ConcurrentMap<String, Session> sessions = new ConcurrentHashMap<>();

    @OnOpen
    public void open(Session session) {
        sessions = sessions.putIfAbsent(session.getId(), session);
    }

    @OnClose
    public void close(Session session) {
        sessions = sessions.remove(session.getId());
    }

}