java - 如何处理 Cassandra 中的竞争条件?

标签 java spring cassandra race-condition spring-data-cassandra

我正在使用 Cassandra 数据库开发一个简单的 Spring Boot 微服务。

我的服务器中的主要实体在下面的 Subscription 类中:

public final class Subscription {

    private String id;

    private String firstUser;

    private String secondUser;

    private String firstUserStatus;

    private String secondUserStatus;

}

我的存储库使用订阅对象并将其持久化。 但在坚持之前,我应该检查一些约束,主要是不能有另一个订阅具有相同的 firstUsersecondUser。 它可能失败的情况非常简单:

  1. 将保留一个订阅
  2. 正在验证表是否存在另一个订阅 相同的用户
  3. 现在可以保留具有相同用户的另一个订阅,并且 使之前的验证过时
  4. 持续订阅。

正确的解决方法是什么?

值得一提的是:用户的顺序并不重要。如果要保留用户“1”和“2”的订阅,则必须存在用户“1”和“2”或“2”和“1”的其他订阅。

UPD:考虑到@bhspencer 的回答,我可以提出另一种情况(我还需要强制状态保持一致):

  1. 将使用 firstUserStatus="ISSUED"secondUserStatus="WAITING_ACTION"
  2. 保留一个订阅
  3. 正在验证表是否存在具有相同用户的另一个订阅
  4. 具有相同 ID(用户)和 firstUserStatus="WAITING_ACTION"secondUserStatus="ISSUED" 的另一个订阅现在可以保留并使之前的验证过时<
  5. 正在保留初始订阅并覆盖已存在的订阅 -> secondUserStatus="ISSUED" 丢失

最佳答案

使对象的 id 成为 firstUser + secondUser 的串联。使 id 成为主键,以确保您的订阅是唯一的。

由于您的排序要求,您应该首先对 firstUser 和 secondUser 进行排序,然后按排序顺序将它们连接起来。这样做意味着 generateId("1", "2") 和 generateId("2". "1") 都会生成相同的 ID“1-2”。

public String generateId(String firstUser, String secondUser) {
  List<String> ids = new ArrayList<>();
  ids.add(firstUser);
  ids.add(secondUser);
  Collections.sort(ids);
  String result = ids.get(0) + "-" + ids.get(1);
  return result;
}

关于java - 如何处理 Cassandra 中的竞争条件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54184281/

相关文章:

java - Spring Boot 集成测试因 java.awt.HeadlessException 失败

java - 为什么 getAnnotatedParameterTypes 看不到数组类型的注释?

spring - 为什么我的 spring boot 应用程序首先失败然后突然在 cloudfoundry 中启动?

java - Spring WS 生成通知而不是请求响应

nosql - 对于 Cassandra 集群,我绝对需要至少 3 个节点/服务器还是 2 个就足够了?

java - 有兴趣学习Hadoop框架

java - 下载带有 HttpURLConnection 问题的 *.deb 文件

java - Spring注解@Retryable——如何设置一个拦截器

java - Kafka与Cassandra数据库连接

cassandra - 我可以将 Apache cassandra 节点添加到 DataStax (DSE) cassandra 集群吗?