保存事件的竞争条件

Race Condition On Save Event

我有两个不同的端点,例如:

endpoint1 => /api/user
endpoint2 => /api/user/roles

假设这两个端点同时调用相同的方法。

public List activeRoles(userId) {
   var roles = repository.findAllRolesOfUser(userId);

   if (roles == null) {
      var role = new Role("DEFAULT", userId);
      repository.save(role);
      
      return Arrays.asList(role)
   } else {
      return roles;
   }
}

当我同时从端点 1 和端点 2 调用 activeRoles(userId) 方法时,它会导致竞争条件,并且会在数据库中创建两次“DEFAULT”角色。有办法避免吗?

注意:我不能使用唯一约束。

在 java.

中使用 syncronized
public List activeRoles(userId) {
   var roles = repository.findAllRolesOfUser(userId);

   if (roles == null) {
      var role = new Role("DEFAULT", userId);
      save(role);
      
      return Arrays.asList(role)
   } else {
      return roles;
   }
}

private syncronized void save(data) {
     var role = repository.findBy(...);
     
     if (!role) {
        repository.save(data);
     }
}