【发布时间】:2023-04-08 20:20:01
【问题描述】:
在 spring-redis-data 中使用多个线程时我遇到了一个大问题,而且它很容易重现,我认为我错过了一些琐碎的事情。
直入主题
如果我在执行保存操作时查询 CrudRepository,有时(高达 60%)在 Redis 上找不到记录。
环境
代码
尽管可以在上面的链接中找到完整的代码,但这是主要组件:
CrudRepository
>@Repository
public interface MyEntityRepository extends CrudRepository<MyEntity, Integer> {
}
实体
@RedisHash("my-entity")
public class MyEntity implements Serializable {
@Id
private int id1;
private double attribute1;
private String attribute2;
private String attribute3;
控制器
@GetMapping( "/my-endpoint")
public ResponseEntity<?> myEndpoint () {
MyEntity myEntity = new MyEntity();
myEntity.setAttribute1(0.7);
myEntity.setAttribute2("attr2");
myEntity.setAttribute3("attr3");
myEntity.setId1(1);
myEntityRepository.save(myEntity);//create it in redis
logger.info("STARTED");
for (int i = 0; i < 100; i++){
new Thread(){
@Override
public void run() {
super.run();
myEntity.setAttribute1(Math.random());
myEntityRepository.save(myEntity); //updating the entity
Optional<MyEntity> optionalMyEntity = myEntityRepository.findById(1);
if (optionalMyEntity.isPresent()) {
logger.info("found");
}else{
logger.warning("NOT FOUND");
}
}
}.start();
}
return ResponseEntity.noContent().build();
}
结果
2020-05-26 07:52:53.769 INFO 30655 --- [nio-8080-exec-2] my-controller-logger : STARTED
2020-05-26 07:52:53.795 INFO 30655 --- [ Thread-168] my-controller-logger : found
2020-05-26 07:52:53.798 WARN 30655 --- [ Thread-174] my-controller-logger : NOT FOUND
2020-05-26 07:52:53.798 WARN 30655 --- [ Thread-173] my-controller-logger : NOT FOUND
2020-05-26 07:52:53.806 INFO 30655 --- [ Thread-170] my-controller-logger : found
2020-05-26 07:52:53.806 WARN 30655 --- [ Thread-172] my-controller-logger : NOT FOUND
2020-05-26 07:52:53.812 WARN 30655 --- [ Thread-175] my-controller-logger : NOT FOUND
2020-05-26 07:52:53.814 WARN 30655 --- [ Thread-176] my-controller-logger : NOT FOUND
2020-05-26 07:52:53.819 WARN 30655 --- [ Thread-169] my-controller-logger : NOT FOUND
2020-05-26 07:52:53.826 INFO 30655 --- [ Thread-171] my-controller-logger : found
2020-05-26 07:52:53.829 INFO 30655 --- [ Thread-177] my-controller-logger : found
所以简单地用 10 个线程,其中 6 个在 db 中找不到结果。
用spring数据redis替换
如here提到的,用spring数据替换redis中的redis至少包含9个操作。
第一个结论
所以,要替换redis中的值,它必须删除哈希,索引,然后再次添加新的哈希和新索引,可能一个线程正在执行此操作而其他线程尝试按索引查找值,该索引尚未添加。
第二个结论
我认为带有data-redis的spring数据几乎不可能有这样的错误,所以我想知道我对data-redis或redis的不了解。由于 redis 具有并发性,我认为可能会发生一些不同的事情,但是对于提供的示例,它似乎是这样的......
提前感谢大家
【问题讨论】:
标签: java spring multithreading concurrency spring-data-redis