wfansh преди 4 месеца
родител
ревизия
525e00cfee

+ 4 - 1
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GACountryReportServiceImpl.java

@@ -9,6 +9,7 @@ import org.jeecg.modules.adweb.dmp.entity.GACountryReport;
 import org.jeecg.modules.adweb.dmp.mapper.GACountryReportMapper;
 import org.jeecg.modules.adweb.dmp.service.IGACountryReportService;
 import org.jeecg.modules.adweb.dmp.vo.report.CountryStatsVO;
+import org.jeecg.modules.redis.CacheConfig;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
@@ -27,7 +28,9 @@ public class GACountryReportServiceImpl extends ServiceImpl<GACountryReportMappe
     @Autowired private GACountryReportMapper gaCountryReportMapper;
 
     @Override
-    @Cacheable(cacheNames = "getCountryStats", key = "{#siteCode, #start, #end}")
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames = "getCountryStats" + 60 * 30) // Redis TTL为半小时
     public List<CountryStatsVO> getCountryStats(String siteCode, Date start, Date end) {
         List<CountryStatsVO> countryStatsVOs =
                 gaCountryReportMapper.getCountryStats(siteCode, start, end);

+ 6 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GADailyReportServiceImpl.java

@@ -16,7 +16,9 @@ import org.jeecg.modules.adweb.dmp.mapper.GADailyReportMapper;
 import org.jeecg.modules.adweb.dmp.service.IGADailyReportService;
 import org.jeecg.modules.adweb.enquiry.mapper.AdwebEnquiryMapper;
 import org.jeecg.modules.adweb.site.service.IAdwebSiteService;
+import org.jeecg.modules.redis.CacheConfig;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.*;
@@ -40,6 +42,10 @@ public class GADailyReportServiceImpl extends ServiceImpl<GADailyReportMapper, G
      *
      * <p>填充每日询盘数据
      */
+    @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames = "getDailyStatsWithinPeriod" + 60 * 30) // Redis TTL为半小时
     public List<DailyStatsVO> getDailyStatsWithinPeriod(String siteCode, Date start, Date end) {
         Map<String, DailyStatsVO> dailyStatsVOs = Maps.newHashMap();
 

+ 6 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GAPagePathReportServiceImpl.java

@@ -14,7 +14,9 @@ import org.jeecg.modules.adweb.dmp.service.IGAPagePathReportService;
 import org.jeecg.modules.adweb.dmp.vo.report.PagePathStatsVO;
 import org.jeecg.modules.adweb.site.entity.AdwebSite;
 import org.jeecg.modules.adweb.site.service.IAdwebSiteService;
+import org.jeecg.modules.redis.CacheConfig;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.Collections;
@@ -34,6 +36,10 @@ public class GAPagePathReportServiceImpl
 
     @Autowired private IAdwebSiteService adwebSiteService;
 
+    @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames = "getPagePathStats" + 60 * 30) // Redis TTL为半小时
     public List<PagePathStatsVO> getPagePathStats(
             String siteCode, Date start, Date end, int limit) {
         List<PagePathStatsVO> pagePathStatsVOs =

+ 5 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GASourceMediumReportServiceImpl.java

@@ -8,7 +8,9 @@ import org.jeecg.modules.adweb.dmp.entity.GASourceMediumReport;
 import org.jeecg.modules.adweb.dmp.mapper.GASourceMediumReportMapper;
 import org.jeecg.modules.adweb.dmp.service.IGASourceMediumReportService;
 import org.jeecg.modules.adweb.dmp.vo.report.SourceMediumStatsVO;
+import org.jeecg.modules.redis.CacheConfig;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.Collections;
@@ -25,6 +27,9 @@ public class GASourceMediumReportServiceImpl
     @Autowired private GASourceMediumReportMapper gaSourceMediumReportMapper;
 
     @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames = "getSourceMediumStats" + 60 * 30) // Redis TTL为半小时
     public List<SourceMediumStatsVO> getSourceMediumStats(String siteCode, Date start, Date end) {
         List<SourceMediumStatsVO> sourceMediumStatsVOs =
                 gaSourceMediumReportMapper.getSourceMediumStats(siteCode, start, end);

+ 86 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/redis/CacheConfig.java

@@ -0,0 +1,86 @@
+package org.jeecg.modules.redis;
+
+import static org.springframework.data.redis.serializer.RedisSerializationContext.SerializationPair;
+
+import com.fasterxml.jackson.annotation.JsonAutoDetect;
+import com.fasterxml.jackson.annotation.PropertyAccessor;
+import com.fasterxml.jackson.databind.ObjectMapper;
+
+import org.jeecg.common.modules.redis.config.RedisConfig;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.CacheManager;
+import org.springframework.cache.annotation.Cacheable;
+import org.springframework.cache.annotation.EnableCaching;
+import org.springframework.cache.interceptor.CacheAspectSupport;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.context.annotation.Primary;
+import org.springframework.data.redis.cache.RedisCacheConfiguration;
+import org.springframework.data.redis.cache.RedisCacheWriter;
+import org.springframework.data.redis.connection.RedisConnectionFactory;
+import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
+import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
+import org.springframework.data.redis.serializer.StringRedisSerializer;
+
+import java.time.Duration;
+
+/**
+ * Redis cache配置
+ *
+ * @author wfansh
+ */
+@Configuration
+@EnableCaching
+public class CacheConfig {
+
+    public static final String PRIMARY_CACHE_MANAGER = "primaryCacheManager";
+
+    public static final String TTL_CACHE_MANAGER = "ttlCacheManager";
+
+    private static final int DEFAULT_TTL_HOURS = 5;
+
+    /** Jeecg默认Redis cache配置 */
+    @Autowired private RedisConfig redisConfig;
+
+    /**
+     * Workaround - 重新注册Jeecg默认的{@link CacheManager},标识为Primary
+     *
+     * <p>当有多个{@link CacheManager} beans时,需标识出@Prinary
+     *
+     * <p>参考{@link RedisConfig#cacheManager(LettuceConnectionFactory)}方法
+     *
+     * <p>参考{@link CacheAspectSupport#setCacheManager(CacheManager)}方法
+     */
+    @Bean(name = PRIMARY_CACHE_MANAGER)
+    @Primary
+    public CacheManager primaryCacheManager(LettuceConnectionFactory connectionFactory) {
+        return redisConfig.cacheManager(connectionFactory);
+    }
+
+    /** 注册{@link CacheManager},支持在{@link Cacheable#cacheNames()}中设置Redis TTL */
+    @Bean(name = TTL_CACHE_MANAGER)
+    public CacheManager ttlCacheManager(RedisConnectionFactory connectionFactory) {
+        ObjectMapper objectMapper = new ObjectMapper();
+        // 解决缓存对象转换异常的问题
+        objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
+        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
+
+        // 配置序列化(解决乱码的问题),并配置缓存默认TTL
+        RedisCacheConfiguration cacheConfiguration =
+                RedisCacheConfiguration.defaultCacheConfig()
+                        .entryTtl(Duration.ofHours(DEFAULT_TTL_HOURS))
+                        .serializeKeysWith(
+                                SerializationPair.fromSerializer(new StringRedisSerializer()))
+                        .serializeValuesWith(
+                                SerializationPair.fromSerializer(
+                                        new Jackson2JsonRedisSerializer(
+                                                objectMapper, Object.class)))
+                        .disableCachingNullValues();
+
+        // 非锁写入方式
+        RedisCacheWriter cacheWriter =
+                RedisCacheWriter.nonLockingRedisCacheWriter(connectionFactory);
+
+        return new TTLCacheManager(cacheWriter, cacheConfiguration);
+    }
+}

+ 55 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/redis/TTLCacheManager.java

@@ -0,0 +1,55 @@
+package org.jeecg.modules.redis;
+
+import static org.springframework.data.redis.serializer.RedisSerializationContext.SerializationPair;
+
+import lombok.extern.slf4j.Slf4j;
+
+import org.apache.commons.lang3.StringUtils;
+import org.jeecg.modules.adweb.dmp.service.impl.GACountryReportServiceImpl;
+import org.springframework.cache.annotation.Cacheable;
+import org.springframework.data.redis.cache.*;
+import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
+
+import java.time.Duration;
+import java.util.Date;
+
+/**
+ * 支持在{@link Cacheable#cacheNames()}中设置Redis TTL,以#分隔,单位为秒
+ *
+ * <p>示例: @Cacheable(cacheManager = CacheConfig.TTL_CACHE_MANAGER, cacheNames = "test:demo#3600")
+ *
+ * <p>用法参考{@link GACountryReportServiceImpl#getCountryStats(String, Date, Date)}方法
+ *
+ * @author wfansh
+ */
+@Slf4j
+public class TTLCacheManager extends RedisCacheManager {
+
+    public TTLCacheManager(
+            RedisCacheWriter cacheWriter, RedisCacheConfiguration defaultCacheConfiguration) {
+        super(cacheWriter, defaultCacheConfiguration);
+    }
+
+    @Override
+    protected RedisCache createRedisCache(String name, RedisCacheConfiguration cacheConfiguration) {
+        // 修改缓存key和value值的序列方式
+        cacheConfiguration =
+                cacheConfiguration
+                        .computePrefixWith(cacheName -> cacheName + ":")
+                        .serializeValuesWith(
+                                SerializationPair.fromSerializer(
+                                        new GenericJackson2JsonRedisSerializer()));
+
+        final int idx = StringUtils.lastIndexOf(name, '#');
+        if (idx > -1) {
+            final String ttl = StringUtils.substring(name, idx + 1);
+            name = StringUtils.substring(name, 0, idx);
+
+            // 设置TTL
+            cacheConfiguration =
+                    cacheConfiguration.entryTtl(Duration.ofSeconds(Long.parseLong(ttl)));
+        }
+
+        return super.createRedisCache(name, cacheConfiguration);
+    }
+}