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??
}
}
答案 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());
}
}