架构师方案-本地缓存一致性刷新方案

前言

本地缓存和业务请求在同一台机器上,相对于Redis读写速度非常快,对于一些变更频率低、实时性要求低的数据,可以放在本地缓存中,提升访问速度。 使用本地缓存能够减少和Redis类集中式缓存间的数据交互,减少网络I/O开销,降低这一过程中在网络通信上的耗时,同时减轻对Redis类集中式缓存访问压力。

解决方案

服务是多节点部署的,要保证是本地缓存一致性的,就要短时间内操作所有服务。

方案1-MQ广播消息

redis发布订阅功能实现同理

方案2-Zookeeper Watcher机制

Tomcat Watcher监听

typescript 复制代码
@Service
public class ZkDataListenerImpl implements IZkDataListener {
    @Override
    public void handleDataChange(String dataPath, Object data) throws Exception {
        String type = getType(data);
        if (type.equals("update")) {
            //更新缓存
       
        }
        if (type.equals("delete")) {
          //删除缓存
        }
    }

}

方案3-RPC框架广播调用(如dubbo广播调用方式)

ini 复制代码
<dubbo:service cluster="broadcast" />

实现逻辑

  1. 循环调用所有的实例
  2. 如果有发生异常则记录异常保存
  3. 只要有异常,则抛出异常,如果没有则返回执行结果

源码

scala 复制代码
public class BroadcastClusterInvoker<T> extends AbstractClusterInvoker<T> {

    private static final Logger logger = LoggerFactory.getLogger(BroadcastClusterInvoker.class);

    public BroadcastClusterInvoker(Directory<T> directory) {
        super(directory);
    }

    @Override
    @SuppressWarnings({"unchecked", "rawtypes"})
    public Result doInvoke(final Invocation invocation, List<Invoker<T>> invokers, LoadBalance loadbalance) throws RpcException {
        // 检查 invokers 是否为空
        checkInvokers(invokers, invocation);

        // 将invokers 塞到context中
        RpcContext.getContext().setInvokers((List) invokers);
        RpcException exception = null;
        Result result = null;

        // 遍历invokers 执行, 结果只要最后一个不报错的result
        for (Invoker<T> invoker : invokers) {
            try {
                result = invoker.invoke(invocation);
            } catch (RpcException e) {
                exception = e;
                logger.warn(e.getMessage(), e);
            } catch (Throwable e) {
                exception = new RpcException(e.getMessage(), e);
                logger.warn(e.getMessage(), e);
            }
        }
        // 如果出现一个异常,  抛出异常
        if (exception != null) {
            throw exception;
        }
        return result;
    }
}

方案4-分布式任务调度的广播执行任务

广播执行表示一个任务实例会广播到该分组所有Worker上执行,当所有Worker都执行完成。

方案对比

方案 特点
MQ广播消息 有消息积压、消息顺序的问题
Zookeeper Watcher机制 Zookeeper本身适合读多写少的场景
RPC框架广播调用(如dubbo广播调用方式) 循环调用所有的实例,所有要考虑实例过多的情况
分布式任务调度的广播执行任务 便于定时发布

总结

从以上这些方案,不难看出,只要有广播功能特点的中间件或服务,都可以用来操作本地缓存或者本地方法。
相关推荐
AWS官方合作商1 小时前
基于AWS无服务器架构的区块链API集成:零基础设施运维实践
架构·serverless·web3·区块链·aws
掘金-我是哪吒2 小时前
分布式微服务系统架构第145集:Jeskson文档-微服务分布式系统架构
分布式·微服务·云原生·架构·系统架构
你怎么知道我是队长3 小时前
GO语言---匿名函数
开发语言·后端·golang
G探险者8 小时前
为什么 Zookeeper 越扩越慢,而 Nacos 却越扩越快?
分布式·后端
不太厉害的程序员8 小时前
NC65配置xml找不到Bean
xml·java·后端·eclipse
不被定义的程序猿8 小时前
Golang 在 Linux 平台上的并发控制
开发语言·后端·golang
AntBlack8 小时前
Python : AI 太牛了 ,撸了两个 Markdown 阅读器 ,谈谈使用感受
前端·人工智能·后端
mikes zhang9 小时前
Flask文件上传与异常处理完全指南
后端·python·flask
Pitayafruit9 小时前
跟着大厂学架构01:如何利用开源方案,复刻B站那套“永不崩溃”的评论系统?
spring boot·分布式·后端
方圆想当图灵10 小时前
深入理解软件设计:领域驱动设计 DDD
后端·架构