当前位置: 首页 > article >正文

分布式本地缓存 ehcache 缓存同步复制

使用spring cache的方式

spring:
  #ehcache 配置
  cache:
    # 指定缓存类型 ehcache 本地缓存 redis 缓存
    type: ehcache
    ehcache:
      config: classpath:ehcache.xml
    redis:
      # 指定存活时间(ms)
      time-to-live: 86400000
      # 指定前缀
      use-key-prefix: true
      # 是否缓存空值,可以防止缓存穿透
      cache-null-values: true

缓存是在本地内存中的,当部署多台实例缓存无法共享缓存,导致登录状态异常等问题。

ehcache 分为2.x和3.x版本,差异比较大,我这使用的是2.10.9.2 版本;
比较常见的同步方式
(1)RMI
rmi 是ehcache本身支持的,不需要引入其他jar包;
(2)JGroups 需要引入其他jar包,配置好后部署会比较方便

<dependency>
<groupId>net.sf.ehcache</groupId>
            <artifactId>ehcache-jgroupsreplication</artifactId>
            <version>1.7</version>
                    </dependency>
<dependency>
            <groupId>org.jgroups</groupId>
            <artifactId>jgroups</artifactId>
            <version>3.6.17.Final</version>
        </dependency>

(3)Terrocotta (4)JMS

我这里用的是RMI的同步方式

系统需要部署两份,机器ip分别为10.59.0.1 和 10.59.0.2
修改项目中ehcache.xml 文件配置
<# red>231

标识当前服务是一个分片,以及需要向哪里发送数据,哪些缓存池需要同步

部署10.59.0.2 机器的时候需要将 rmiUrls 改为 0.1:40000

<cacheManagerPeerProviderFactory     class="net.sf.ehcache.distribution.RMICacheManagerPeerProviderFactory" properties="peerDiscovery=manual,rmiUrls=//10.59.0.2:40000/localTempSwap
            |//10.59.0.2:40000/temp_cache"/>
# cacheManagerPeerProviderFactory 标识服务是集群中的一份子,配置数据写入目的地,
#peerDiscovery=manual 手动发现其他服务;
#rmiUrls=//10.59.0.2:40001/localTempSwap|//10.59.0.2:40001/temp_cache    数据变更时候像这个端口的cache池写数据,多个用| 分隔;ip 写上其他机器的ip, 端口对应cacheManagerPeerListenerFactory 标签中的监听端口port;

当前服务ip 端口信息

部署10.59.0.2 机器的时候需要将 hostName 改为 0.2


<cacheManagerPeerListenerFactory
            class="net.sf.ehcache.distribution.RMICacheManagerPeerListenerFactory"
            properties="hostName=10.59.0.1,port=40000,remoteObjectPort=50000"/>
#            位于ehcache标签内
#hostName  本机ip, port监听端口, remoteObjectPort发送数据时候所使用端口

还需要在每个cache 中配置哪些动作需要触发同步

<cacheEventListenerFactory class="net.sf.ehcache.distribution.RMICacheReplicatorFactory"
                                   properties="replicateAsynchronously=false, replicatePuts=true, replicateUpdates=true,
                 replicateUpdatesViaCopy=true, replicateRemovals=true "/>
#配置在每一个需要同步的<cache>标签中
#replicateUpdatesViaCopy:
        是否将对象变更复制到所有节点,还是只是发送一个失效信息,让对方该缓存失效,当对方需要该缓存时重新计算载入。
        默认为true。鉴于对象复制的消耗挺大的,又有锁的问题,而且对方也未必需要该对象,所以此属性建议设为false。
        如果业务上真的需要设为true时,就可考虑使用Terracotta了。
        
        #replicatePuts:
        增加对象时是否同步,默认为true,如果replicateUpdatesViaCopy为false,选择了失效算法,所以replicatePuts 要设为false。
        
        #replicateUpdates:
        修改加对象时是否同步,默认为true
        
        #replicateRemovals:
        删除加对象时是否同步,默认为true`

附带一个完整的xml 配置

<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd">

    <!-- 磁盘缓存位置 -->
    <diskStore path="java.io.tmpdir"/>

    <cacheManagerPeerProviderFactory
            class="net.sf.ehcache.distribution.RMICacheManagerPeerProviderFactory"
            properties="peerDiscovery=manual,rmiUrls=//10.59.0.2:40000/demo"/>
    <cacheManagerPeerListenerFactory
            class="net.sf.ehcache.distribution.RMICacheManagerPeerListenerFactory"
            properties="hostName=10.59.0.1,port=40000,remoteObjectPort=50000"/>

    <!-- 默认缓存 -->
    <defaultCache
            maxElementsInMemory="10000"
            eternal="false"
            timeToIdleSeconds="120"
            timeToLiveSeconds="120"
            maxElementsOnDisk="10000000"
            diskExpiryThreadIntervalSeconds="120"
            memoryStoreEvictionPolicy="LRU">
        <persistence strategy="localTempSwap"/>
        <cacheEventListenerFactory class="net.sf.ehcache.distribution.RMICacheReplicatorFactory"
                                   properties="replicateAsynchronously=false, replicatePuts=true, replicateUpdates=true,
                 replicateUpdatesViaCopy=true, replicateRemovals=true "/>
    </defaultCache>

</ehcache>


http://www.kler.cn/a/302152.html

相关文章:

  • Python——day09
  • 24 go语言(golang) - gorm框架安装及使用案例详解
  • 云手机+YouTube:改变通信世界的划时代技术
  • GIT与github的链接(同步本地与远程仓库)
  • VSCode 性能优化指南:提高编码效率,减少资源占用
  • 大恒相机开发(2)—Python软触发调用采集图像
  • Javaweb项目实现文件导出功能
  • 服务器数据恢复—Linux操作系统环境下网站数据的恢复案例
  • Java、python、php、node.js版 铁路售票自动选座系统 高铁购票系统 火车订票平台(源码、调试、LW、开题、PPT)
  • Android Graphics 显示系统 - VirtualDisplay的初印象 - 简单示例
  • .Net 中各种线程同步锁
  • Gitea Action 简单配置(CI/CD)
  • java 学习从零到精通之历程
  • C4D2025来了!亮眼的新功能一览
  • 高亚科技与广东海悟携手,打造全流程电子竞标管理平台!
  • 《程序猿之设计模式实战 · 策略模式》
  • 深度解读MySQL意向锁的工作原理机制与应用场景
  • 使用Selenium与WebDriver实现跨浏览器自动化数据抓取
  • 信息安全工程师(1)计算机网络分类
  • Linux Makefile文本处理函数知识详解
  • 【Http 每天一小问 ,Post上传文件时, 文件和 -d(--data)和 -F(--form) 不能同时存在 ,怎么办】
  • Linux系统安装CUDA
  • JSON对象
  • JSDelivr NPM CDN 国内加速节点
  • java设计模式(持续更新中)
  • 英伟达与甲骨文携手加速企业级AI和数据处理