SpringBoot集成Redis,并自定义对象序列化

SpringBoot集成Redis,并自定义对象序列化SpringBoot项目使用redis非常简单,pom里面引入redis的场景启动器,在启动类上加@EnableCaching注解,项目启动会自动匹配上redis,这样项目中就可以愉快地使用了,使用方法:要么使用@Cacheable一类的注解自动缓存,要么使用RedisTemplate手动缓存。(前提是你的本机或者是远程主机要先搭好redis环境)虽然SpringBoot好用,但这里也有好多…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

SpringBoot项目使用redis非常简单,pom里面引入redis的场景启动器,在启动类上加@EnableCaching注解,项目启动会自动匹配上redis,这样项目中就可以愉快地使用了,
使用方法:要么使用@Cacheable一类的注解自动缓存,要么使用RedisTemplate手动缓存。
前提是你的本机或者是远程主机要先搭好redis环境
虽然SpringBoot好用,但这里也有好多坑,SpringBoot和MySQL一样,易学难精,阳哥说的对,练武不练功,到老一场空。
下面,我将详细记录整合步骤

  1. 创建一个SpringBoot项目
    pom文件中引入cache和redis依赖
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-cache</artifactId>
</dependency>
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

2.新建cache数据库,创建员工表employee,写好DAO,service,controller,配置文件application.yml中配好数据源和redis地址

spring:
  datasource:
    driver-class-name: com.mysql.jdbc.Driver
    url: jdbc:mysql://localhost:3305/cache
    username: root
    password: 123456
  redis:
    host: 127.0.0.1
    port: 6379
    password: hehuiredis177
    
mybatis:
  configuration:
    map-underscore-to-camel-case: true
  mapper-locations:
  - classpath:mybatis/mapper/*.xml
  
debug: true

3.SpringBoot主启动类上加@EnableCaching注解,service类@Cacheable(cacheNames=“employee”)注解

@Service
public class EmployeeService {

	@Autowired private EmployeeDao employeeDao;
	
	@Cacheable(cacheNames="employee")
	public List<Employee> findAll(){
		System.out.println("查询所有员工......");
		return employeeDao.findAll();
	}
	
	@Cacheable(cacheNames="employee--id")
	public Employee selectEmp(Integer id){
		System.out.println("查询员工:"+id);
		return employeeDao.selectById(id);
	}
	
}

3.启动项目(application.yml加上debug:true可以将自动配置报告打印在控制台方便观察)
在这里插入图片描述
项目启动时,自动配置报告提示Redis已经matched(这里说一下,你如果没有引入redis而又在启动类上加了@EnableCaching注解,SpringBoot会自动给你匹配一个SimpleCacheConfiguration缓存,它的底层用了一个key-value的Map,不能像redis一样持久化,您得注意,好好权衡一下用不用,笔者不建议使用,像SQLite、H2这些一样,玩具型的,只适合个人博客等非正式场合使用,有轻量级的优点,也有不可靠,不好管理的缺点)

4.然后访问 http://localhost:8080/emp/1
在这里插入图片描述
500错误,服务端有问题,仔细阅读报错信息,是实体类没有序列化导致的,好吧,那我就实现Serializable接口

5.练武不练功,到老一场空,实现序列化Serializable就可以了吗?是可以了,但这只是巧合,再去访问,确实可以
在这里插入图片描述
去看redis
在这里插入图片描述
什么东西啊,看不懂…
在这里插入图片描述
但第二次以后确实走了缓存,离成功又近了一步
再去试一下查所有
在这里插入图片描述
也可以
缓存也有
在这里插入图片描述
6.为什么实现Serializable接口就可以了,这就得开始扒源码了,开始练功…
去看RedisTemplate这个类
在这里插入图片描述
对象默认实现序列化
怎么实现的呢?
接着往下看
在这里插入图片描述
默认用的是jdk的序列化机制
所以实现Serializable接口就可以,巧合!!!

7.继续练功,redis里这么乱,看不懂,这显然不是我们想要的,我们要的是json,轻量易读的json才是我们的目标,欲存json,必须改变序列化机制,把jdk的序列化替换掉
继续读源码
找到RedisAutoConfiguration这个类,看它是怎么自动配置的
里面有个内部类RedisConfiguration
在这里插入图片描述
初始化的时候,把RedisTemplate和StringRedisTemplate放在了IOC容器里,也就是说我们可以在我们自己的代码里注入这两个类,然后手动缓存
而@Cacheable注解使用的是RedisTemplate,那么分析一下RedisTemplate

		@Bean
		@ConditionalOnMissingBean(name = "redisTemplate")
		public RedisTemplate<Object, Object> redisTemplate(
				RedisConnectionFactory redisConnectionFactory)
				throws UnknownHostException {
			RedisTemplate<Object, Object> template = new RedisTemplate<Object, Object>();
			template.setConnectionFactory(redisConnectionFactory);
			return template;
		}

RedisTemplate<Object, Object> template = new RedisTemplate<Object, Object>();
直接new了一个RedisTemplate然后把它返回,呵呵!就从这里入手
@ConditionalOnMissingBean(name = “redisTemplate”)表示我们可以自定义RedisTemplate,我们有RedisTemplate,就用我们的,否则就用默认,我们不喜欢你的,要json就要自己配一个RedisTemplate
自己写

package com.hehui.config;

import java.net.UnknownHostException;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;

/**
 * 自定义redis配置
 * @author hehui 
 * @date 2019年3月12日
 *
 */
@Configuration
public class MyRedisConfig {

	@Bean
	public RedisTemplate<Object, Object> redisTemplate(
			RedisConnectionFactory redisConnectionFactory)
			throws UnknownHostException {
		RedisTemplate<Object, Object> template = new RedisTemplate<Object, Object>();
		template.setConnectionFactory(redisConnectionFactory);
		Jackson2JsonRedisSerializer<Object> serializer = new Jackson2JsonRedisSerializer<>(Object.class);
		template.setDefaultSerializer(serializer);
		return template;
	}
	
}

Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer<>(Object.class);
template.setDefaultSerializer(serializer);
只多了这两行代码,其它什么都没变。
为什么要用Jackson2JsonRedisSerializer呢?
因为Jackson2JsonRedisSerializer实现了RedisSerializer接口
在这里插入图片描述
8.再访问 http://localhost:8080/emp/1
在这里插入图片描述
观察redis
在这里插入图片描述
key是我们要的key,value是我们要的value
再刷新页面
在这里插入图片描述
不会吧!!!
去缓存找,找到了,反序列化失败…
转换异常,看来不能用Jackson2JsonRedisSerializer,换成GenericJackson2JsonRedisSerializer
换了之后,刷新页面
在这里插入图片描述
报错信息变了,这是不能读取json,现在redis里的json是Jackson2JsonRedisSerializer序列化的,GenericJackson2JsonRedisSerializer当然不认识咯,清了redis再试
在这里插入图片描述
好了。

9.试试查所有 http://localhost:8080/list
在这里插入图片描述
报错!!!
SimpleKey and no properties discovered 没有key 什么情况?
于是继续读源码,看key是咋生成的

	/**
	 * Generate a key based on the specified parameters.
	 */
	public static Object generateKey(Object... params) {
		if (params.length == 0) {
			return SimpleKey.EMPTY;
		}
		if (params.length == 1) {
			Object param = params[0];
			if (param != null && !param.getClass().isArray()) {
				return param;
			}
		}
		return new SimpleKey(params);
	}

是根据参数来的,我查询所有,是没有参数的,这样key就为空
if (params.length == 0) {

return SimpleKey.EMPTY;
}
这怎么行,开发中,不传参的方法多的很,这样生成key肯定不行撒
于是重写key的生成策略,用【类名+方法名+参数名】这样就可以保证key不为空

package com.hehui.config;

import java.lang.reflect.Method;
import java.net.UnknownHostException;
import java.util.Arrays;

import org.springframework.cache.annotation.CachingConfigurerSupport;
import org.springframework.cache.interceptor.KeyGenerator;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;

/**
 * 自定义redis配置
 * @author hehui 
 * @date 2019年3月12日
 *
 */
@Configuration
public class MyRedisConfig extends CachingConfigurerSupport{

	@Bean
	public RedisTemplate<Object, Object> redisTemplate(
			RedisConnectionFactory redisConnectionFactory)
			throws UnknownHostException {
		RedisTemplate<Object, Object> template = new RedisTemplate<Object, Object>();
		template.setConnectionFactory(redisConnectionFactory);
//		Jackson2JsonRedisSerializer<Object> serializer = new Jackson2JsonRedisSerializer<>(Object.class);
		GenericJackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer();
		template.setDefaultSerializer(serializer);
		return template;
	}

	@Bean
	@Override
	public KeyGenerator keyGenerator() {
		return new KeyGenerator() {
            public Object generate(Object target, Method method, Object... objects) {
                StringBuilder sb = new StringBuilder();
                sb.append(target.getClass().getName()).append(".").append(method.getName()).append(Arrays.toString(objects));
                return sb.toString();
            }
        };
	}
	
	
	
}

extends CachingConfigurerSupport并重写keyGenerator方法
测试通过
在这里插入图片描述
redis
在这里插入图片描述
json中还带有@Class相当好。

10.好啦,相当曲折地把redis整合并自定义配置到SpringBoot中。

源码github地址:https://github.com/hehuihh/springboot_redis

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/190784.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • p2p在线直播流(何为流媒体)

    看到网上一些吹牛P2P低延时的文章,觉得不是很靠谱,抽空调研了一下这个问题。P2P低延时的几个方向:   方法一:通过直接采集并编码多媒体帧,将多媒体帧切分成1KB大小的数据颗粒,采用push策略的进行小包传输,提高传输效率,减小传输延时;          具体参见:http://www.google.com/patents/CN101945129A?cl

    2022年4月10日
    69
  • 信息搜集 – 二层发现 arping[通俗易懂]

    信息搜集 – 二层发现 arping[通俗易懂]0x00:简介在被动信息搜集工作完成后,需要在进一步的对目标进行主动信息搜集,这一阶段主要搜索的信息包括目标主机是否存活,上面开放了哪些端口,有哪些服务,服务系统是什么,开发服务的版本以及上面支撑系统运行的一些中间件或者其他软件的版本(后续可根据版本号查看是否有公开的漏洞问题),在目标主机发现的过程中,不仅要发现目标是否存活,还要发现其整个网段下的其他设备,同时,这些其他设备也应该像目标一样搜…

    2022年5月30日
    39
  • win10下python环境变量设置

    win10下python环境变量设置我用的是python_2.7.3.msi,从官网下载之后,一路按照默认进行安装。安装之后配置环境变量的步骤如下:1,点“我的电脑”,右键选“属性”2,选择“高级系统设置”>选“环境变量”

    2022年7月5日
    25
  • 游戏辅助脚本论坛_开心躲猫猫穿墙版下载

    游戏辅助脚本论坛_开心躲猫猫穿墙版下载总结写开心外挂的心得对开心网的外挂编写做个总结,避免弯路。设计篇         针对需求的设计——需求向实现的转化             从调查切入点开始考虑如何着手,确定了基本的方式到写代码,并没有花太多精力,http访问、html解析都不复杂,但后续需求考虑不足造成最初的设计不能满足后续需要。比如后续加入的新模块轮询检索,定时执行都推翻了最初的设计。在设计阶…

    2022年9月13日
    5
  • 网络交换机光口和电口_交换机的光口

    网络交换机光口和电口_交换机的光口 一、光口1、基本概念     光口是光纤接口的简称。   也可称之为:G口 (意思是G光纤口)   光口:所应用于机房,机柜等大型设备的一个光纤带宽接口。   光纤可以用于音频(声卡有光输出的),网络(光纤作为传输介质),磁盘(光纤代替电缆传输数据)等等。   光纤又可分为单模光纤和多模光纤区别如下:   单模光纤和多模光纤可以从纤芯的尺…

    2022年10月21日
    2
  • c++中vector向量几种情况的总结(向量指针,指针的向量)

    c++中vector向量几种情况的总结(向量指针,指针的向量)1.标准库vector类型vector是同一种类型的对象的集合。每一个对象都有一个相应的整数索引值。标准库将负责管理与存储元素相关的内存。我们把vector称为容器,是由于它能够包括其它对象。一个容器中的全部对象都必须是同一种类型的。用vector之前,必须包括对应的头文件。#include<vector>usingstd::vector;vec…

    2022年6月15日
    24

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号