情况:后端涉及到异步操作,数据还没更新完就直接向前端返回success的结果。

情况:后端涉及到异步操作,那么可能数据还没更新完就直接向前端返回success的结果。这就导致数据不一致。因此需要采用锁或者其他机制,对异步场景进行处理,确保数据的一致性。

下面是一些常用的方法,简单情况直接用CountDownLatch计数器锁即可实现,下面还有多种方法,包括回调,消息队列等。

下面有代码的示例,大家可以根据需要,自行查看!

1. CountDownLatch(计数器锁)

适用于简单的多线程同步场景,等待一个或多个线程完成操作后再继续执行。

java 复制代码
@PostMapping("syncData")
public AjaxJson syncData() {
    CountDownLatch latch = new CountDownLatch(1); // 初始化计数器为1
    
    // 异步操作
    asyncService.processData(latch);
    
    try {
        // 等待最多5秒
        if (!latch.await(5, TimeUnit.SECONDS)) {
            return AjaxJson.getError("操作超时");
        }
    } catch (InterruptedException e) {
        Thread.currentThread().interrupt();
        return AjaxJson.getError("操作被中断");
    }
    
    return AjaxJson.getSuccess();
}
​
// 异步服务中
public void processData(CountDownLatch latch) {
    try {
        // 执行异步操作...
    } finally {
        latch.countDown(); // 操作完成,计数器减1
    }
}

2. CompletableFuture(Java 8+)

更现代的异步编程方式,支持链式调用和组合多个异步操作。

java 复制代码
@PostMapping("syncData")
public CompletableFuture<AjaxJson> syncData() {
    return CompletableFuture.supplyAsync(() -> {
        // 异步操作
        return processData();
    }).thenApply(result -> {
        return AjaxJson.getSuccessData(result);
    }).exceptionally(ex -> {
        return AjaxJson.getError("操作失败: " + ex.getMessage());
    }).completeOnTimeout(
        AjaxJson.getError("操作超时"), 
        5, TimeUnit.SECONDS
    );
}
​
private SomeResult processData() {
    // 执行耗时操作...
    return someResult;
}

3. 回调机制

适用于事件驱动的架构,通过回调函数处理异步结果。

java 复制代码
@PostMapping("syncData")
public AjaxJson syncData() {
    CompletableFuture<Boolean> future = new CompletableFuture<>();
    
    asyncService.processData(new AsyncCallback() {
        @Override
        public void onSuccess() {
            future.complete(true);
        }
        
        @Override
        public void onFailure(Exception e) {
            future.completeExceptionally(e);
        }
    });
    
    try {
        Boolean result = future.get(5, TimeUnit.SECONDS);
        return result ? AjaxJson.getSuccess() : AjaxJson.getError();
    } catch (Exception e) {
        return AjaxJson.getError(e.getMessage());
    }
}

4. 消息队列+数据库状态

适用于分布式系统,通过消息队列和数据库状态保证最终一致性。

java 复制代码
@PostMapping("startSync")
public AjaxJson startSync() {
    String taskId = UUID.randomUUID().toString();
    // 初始化任务状态为"处理中"
    taskService.createTask(taskId, "PROCESSING");
    
    // 发送消息到队列
    messageQueue.send(new SyncMessage(taskId));
    
    return AjaxJson.getSuccessData(taskId);
}
​
@GetMapping("checkStatus/{taskId}")
public AjaxJson checkStatus(@PathVariable String taskId) {
    Task task = taskService.getTask(taskId);
    if ("COMPLETED".equals(task.getStatus())) {
        return AjaxJson.getSuccess();
    } else if ("FAILED".equals(task.getStatus())) {
        return AjaxJson.getError(task.getErrorMessage());
    }
    return AjaxJson.getError("处理中,请稍后再试");
}
​
// 消费者端
@RabbitListener(queues = "syncQueue")
public void processSyncMessage(SyncMessage message) {
    try {
        // 处理业务逻辑...
        taskService.updateTask(message.getTaskId(), "COMPLETED", null);
    } catch (Exception e) {
        taskService.updateTask(message.getTaskId(), "FAILED", e.getMessage());
    }
}

5. 响应式编程(WebFlux)

使用Spring WebFlux进行非阻塞式编程。

java 复制代码
@PostMapping("/syncData")
public Mono<AjaxJson> syncData() {
    return Mono.fromCallable(() -> {
        // 模拟耗时操作
        Thread.sleep(2000);
        return "处理结果";
    })
    .timeout(Duration.ofSeconds(5))
    .map(result -> AjaxJson.getSuccessData(result))
    .onErrorResume(ex -> Mono.just(AjaxJson.getError(ex.getMessage())));
}

最佳实践建议

  1. 简单场景:使用CountDownLatch或CompletableFuture

  2. 复杂异步流程:使用CompletableFuture的组合操作

  3. 分布式系统:使用消息队列+数据库状态

  4. 高并发系统:考虑响应式编程(WebFlux)

  5. 超时处理:所有方案都应设置合理的超时时间

  6. 错误处理:确保异常情况能被妥善处理并反馈给客户端

相关推荐
ps酷教程2 小时前
Jackson 解决没有无参构造函数的反序列化问题
java
NiceCloud喜云2 小时前
Opus 4.8 的 Effort Control 怎么选:Low 到 Max 五档策略
android·java·大数据·前端·c++·python·spring
为思念酝酿的痛2 小时前
POSIX信号量
linux·运维·服务器·后端
小羊在睡觉2 小时前
力扣84. 柱状图中最大的矩形
后端·算法·leetcode·golang·go
ylscode2 小时前
PureLogs 信息窃取恶意软件惊现高危变种:借道 MsBuild.exe 进程空心化实施无痕攻击
网络·安全·安全威胁分析
_日拱一卒3 小时前
LeetCode:994腐烂的橘子
java·数据结构·算法·leetcode·深度优先
云安全助手3 小时前
2026年企业级Claude中转服务深度评测:安全、稳定与速度的终极答案
人工智能·安全·claude·ai大模型
swipe3 小时前
Neo4j + Graph RAG 医疗知识图谱工程实践:患者教育问答真正需要的是“关系可追溯”
后端·langchain·llm
隔窗听雨眠3 小时前
Nginx网关响应慢排查手记
java·服务器·nginx
智慧物业老杨3 小时前
智慧物业合同周期管理系统:从风险预警到智能交接的全流程数智化落地方案
java·人工智能·python