Jetcache开启FASTJSON2序列化

为什么要用Jetcache

JetCache是一个基于Java的缓存系统封装,它提供统一的API和注解来简化缓存的使用。JetCache比SpringCache更强大的注解,可以原生的支持TTL、两级缓存、分布式自动刷新,还提供了Cache接口用于手工缓存操作。

以前使用红薯大佬的j2cache,用起来挺舒服。但是由于j2cache很久没更新了,jedis驱动也比较旧,且不支持TTL。故而转为使用jetcache。

Jetcache现状

目前使用的版本是最新版2.7.4,按照官方的说法,最新版本已经不支持json序列化。但是由于某些数据监控和测试的需要,需要开启json以便在redis客户端能够很方便的观察数据。

有文章说未支持一级缓存的同步。从当前的版本看来,已经是实现了本地缓存的同步了,如果是redis作为二级的话,默认会使用LettuceBroadcastManager进行监听和推送。

需要配置地方

jetcache:
  remote:
    default:
      type: redis.lettuce
      keyConvertor: fastjson2 #other choose:fastjson/jackson
      broadcastChannel: ccframe
      valueEncoder: fastjson2 #other choose:kryo/kryo5/java
      valueDecoder: fastjson2 #other choose:kryo/kryo5/java

只要配置好broadcastChannel即可,然后在生成cache时,指定syncLocal来监听变化:

	@PostConstruct
	public void initCache(){
		QuickConfig qc = QuickConfig.newBuilder(entityClass.getSimpleName() + "Cache")
				.expire(Duration.ofMinutes(30))
				.cacheType(CacheType.BOTH) // two level cache
				.syncLocal(true) // invalidate local cache in all jvm process after update
				.build();
		cache = cacheManager.getOrCreateCache(qc);
	}

此处配置了一个30分钟时长的缓存,发生变更时会广播。

扩展点

回到正题,讲一下如何进行扩展。github上有人扩展了jetcache-plus添加了jackson,需要的可以看看。这里只讲讲如何把jetcache自带的fastjson2给开启起来:

STEP1

按照官方的例子,继承DefaultEncoderParser添加一个新的EncoderParser

package org.ccframe.commons.helper;

import com.alicp.jetcache.CacheConfigException;
import com.alicp.jetcache.anno.SerialPolicy;
import com.alicp.jetcache.anno.support.DefaultEncoderParser;
import com.alicp.jetcache.support.*;
import io.github.qy8502.jetcacheplus.JacksonValueDecoder;
import io.github.qy8502.jetcacheplus.JacksonValueEncoder;
import org.apache.lucene.search.highlight.DefaultEncoder;

import java.net.URI;
import java.util.function.Function;

public class JsonEncoderParser extends DefaultEncoderParser {


    public static final String SERIAL_POLICY_FASTJSON2 = "FASTJSON2";

    @Override
    public Function<Object, byte[]> parseEncoder(String valueEncoder) {

        if (SERIAL_POLICY_FASTJSON2.equalsIgnoreCase(valueEncoder)) {
            return new Fastjson2ValueEncoder(true);
        } else {
            return super.parseEncoder(valueEncoder);
        }
    }

    @Override
    public Function<byte[], Object> parseDecoder(String valueDecoder) {

        if (SERIAL_POLICY_FASTJSON2.equalsIgnoreCase(valueDecoder)) {
            Fastjson2ValueDecoder fastjson2ValueDecoder = new Fastjson2ValueDecoder(true);
            DecoderMap decoderMap = DecoderMap.defaultInstance();
            decoderMap.register(SerialPolicy.IDENTITY_NUMBER_FASTJSON2, Fastjson2ValueDecoder.INSTANCE);
            fastjson2ValueDecoder.setDecoderMap(decoderMap);
            return fastjson2ValueDecoder;
        } else {
            return super.parseDecoder(valueDecoder);
        }
    }

}

至于为什么要用decoderMap.register,因为跟踪代码发现NPE了,翻了下源码参考下,做了个按ID缓存的cache,添加后正常了

STEP2

jetcache设计了一个扩展点,在生成ConfigProvider时,可以指定EncoderParser的bean

/**
 * Created on 2022/08/03.
 */
package com.alicp.jetcache.anno.support;

import com.alicp.jetcache.SimpleCacheManager;
import com.alicp.jetcache.support.StatInfo;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Bean;

import java.util.function.Consumer;

/**
 * used in non-spring-boot projects.
 *
 * @author <a href="mailto:areyouok@gmail.com">huangli</a>
 */
public class JetCacheBaseBeans {

    protected SpringConfigProvider createConfigProvider() {
        return new SpringConfigProvider();
    }

    @Bean(destroyMethod = "shutdown")
    public SpringConfigProvider springConfigProvider(
            @Autowired ApplicationContext applicationContext,
            @Autowired GlobalCacheConfig globalCacheConfig,
            @Autowired(required = false) EncoderParser encoderParser,
            @Autowired(required = false) KeyConvertorParser keyConvertorParser,
            @Autowired(required = false) Consumer<StatInfo> metricsCallback) {
        SpringConfigProvider cp = createConfigProvider();
        cp.setApplicationContext(applicationContext);
        cp.setGlobalCacheConfig(globalCacheConfig);

        if (encoderParser != null) {
            cp.setEncoderParser(encoderParser);
        }

        if (keyConvertorParser != null) {
            cp.setKeyConvertorParser(keyConvertorParser);
        }

        if (metricsCallback != null) {
            cp.setMetricsCallback(metricsCallback);
        }
        cp.init();
        return cp;
    }

    @Bean(name = "jcCacheManager",destroyMethod = "close")
    public SimpleCacheManager cacheManager(@Autowired ConfigProvider configProvider) {
        SimpleCacheManager cacheManager = new SimpleCacheManager();
        cacheManager.setCacheBuilderTemplate(configProvider.getCacheBuilderTemplate());
        return cacheManager;
    }
}

同理keyConvertorParser和metricsCallback也可以类似指定。

因此,只需在connfig类里添加一行代码即完成扩展

@EnableCreateCacheAnnotation            //缓存基本数据据类型的注解
@Configuration
public class RedisConfig{

	@Bean
	public EncoderParser encoderParser(){
		return new JsonEncoderParser();	// 支持json序列化
	}


}

看下redis,已经是json格式了

收工~

Redis中使用FastJson2JsonRedisSerializer进行序列化是一种常见的做法。通过配置RedisConfig类,可以将FastJson2JsonRedisSerializer作为RedisTemplate的序列化器。在RedisConfig类中,通过@Bean注解创建了一个RedisSerializer的实例,并将其设置为RedisTemplate的keySerializer、valueSerializer、hashKeySerializer和hashValueSerializer。这样就可以使用FastJsonRedis中的数据进行序列化和反序列化了。同时,还需要注意在使用@Cacheable注解时,key属性只能传入String类型的值,因为使用了StringRedisSerializer作为key的序列化器。\[2\]\[3\] #### 引用[.reference_title] - *1* [Redis使用FastJson序列化/FastJson2JsonRedisSerializer](https://blog.csdn.net/moshowgame/article/details/83246363)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [SpringBoot之redis使用FastJSON自定义序列化](https://blog.csdn.net/qq_37892957/article/details/89303942)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [阿里FastJson2JsonRedisSerializer.java作为内部类强化RedisConfig的序列化实现](https://blog.csdn.net/as4589sd/article/details/104214206)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值