springboot框架之redis(单机+集群)实现springCache(缓存)
mhr18 2024-11-10 09:49 29 浏览 0 评论
前言:
本文纯手打,为了方便大家亲手操作,所以包含大量源代码,手机阅读请多包涵!
上回给大家分享了springboot框架之redis(单机+集群)实现mybatis缓存,上次的缓存是基于dao持久层的缓存本次我带着大家来搭建一个基于sercice层的缓存,对于缓存放在dao层还是service层,这个问题众说纷纭,我本人倾向于放到service层,因为从软件解耦的观点来说,dao层的实现技术并不局限于mybatis,还有可能是springJDBC,hibrenate等,所以本次我们在service层上加入缓存。
1. redis(单机版)实现spring缓存
1.1. 环境准备
使用maven搭建一套springBoot框架,并创建测试类查询userinfo表。测试代码如下:
/**
* @不开启springCache
*/
@RequestMapping("/findAllUserByJson")
@ResponseBody
public List<Map> findAllUserByJson() {
long start=System.currentTimeMillis();
List<Map> findAllUser = userInfoBizImpl.findAllUser();
long end=System.currentTimeMillis();
System.out.println("查询时间:"+(end-start));
// 将findAllUser转成json字符串
return findAllUser;
}
结果截图:
1.2. 配置spring缓存
1) 修改配置文件pom.xml
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
<version>1.5.8.RELEASE</version>
</dependency>
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.9.0</version>
</dependency>
2) 修改RedisConfig配置类
/**
* 配置redis模版 ,来操作redis服务器
*TODO
*@return
*2018-7-11下午4:06:14
*/
@Bean(name = "redisTemplate")
public RedisTemplate getRedisTemplate(){
RedisTemplate template = new StringRedisTemplate(getJedisConnectionFactory());
return template;
}
/**
* 配置springcacheManager 缓存管理器
*TODO
*@return
*2018-7-11下午4:09:54
*/
@Bean
public SimpleCacheManager getMyCacheManager() {
SimpleCacheManager simpleCacheManager = new SimpleCacheManager();
Set<Cache> caches= new HashSet<Cache>();
// 初始化cache
SpringRedisCache cache = new SpringRedisCache();
cache.setName("chenjian");
cache.setRedisTemplate(getRedisTemplate());
caches.add(cache);
simpleCacheManager.setCaches(caches);
return simpleCacheManager;
}
SpringRedisCache实现org.springframework.cache.Cache;接口
package com.aaa.util;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.concurrent.Callable;
import org.springframework.cache.Cache;
import org.springframework.cache.support.SimpleValueWrapper;
import org.springframework.dao.DataAccessException;
import org.springframework.data.redis.connection.RedisConnection;
import org.springframework.data.redis.core.RedisCallback;
import org.springframework.data.redis.core.RedisTemplate;
/**
* @author TeacherChen
* @description springcache的实现类
* @company AAA软件
* 2018-1-13下午2:07:15
*/
public class SpringRedisCache implements Cache {
private RedisTemplate<String, Object> redisTemplate;
private String name;
public RedisTemplate<String, Object> getRedisTemplate() {
return redisTemplate;
}
public void setRedisTemplate(RedisTemplate<String, Object> redisTemplate) {
this.redisTemplate = redisTemplate;
}
public void setName(String name) {
this.name = name;
}
@Override
public String getName() {
// TODO Auto-generated method stub
return this.name;
}
@Override
public Object getNativeCache() {
// TODO Auto-generated method stub
return this.redisTemplate;
}
@Override
public ValueWrapper get(Object key) {
// TODO Auto-generated method stub
System.out.println("get key");
final String keyf = key.toString();
Object object = null;
object = redisTemplate.execute(new RedisCallback<Object>() {
public Object doInRedis(RedisConnection connection)
throws DataAccessException {
byte[] key = keyf.getBytes();
byte[] value = connection.get(key);
if (value == null) {
return null;
}
return toObject(value);
}
});
return (object != null ? new SimpleValueWrapper(object) : null);
}
@Override
public void put(Object key, Object value) {
// TODO Auto-generated method stub
System.out.println("put key");
final String keyf = key.toString();
final Object valuef = value;
final long liveTime = 86400;
redisTemplate.execute(new RedisCallback<Long>() {
public Long doInRedis(RedisConnection connection)
throws DataAccessException {
byte[] keyb = keyf.getBytes();
byte[] valueb = toByteArray(valuef);
connection.set(keyb, valueb);
if (liveTime > 0) {
connection.expire(keyb, liveTime);
}
return 1L;
}
});
}
private byte[] toByteArray(Object obj) {
byte[] bytes = null;
ByteArrayOutputStream bos = new ByteArrayOutputStream();
try {
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(obj);
oos.flush();
bytes = bos.toByteArray();
oos.close();
bos.close();
} catch (IOException ex) {
ex.printStackTrace();
}
return bytes;
}
private Object toObject(byte[] bytes) {
Object obj = null;
try {
ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
ObjectInputStream ois = new ObjectInputStream(bis);
obj = ois.readObject();
ois.close();
bis.close();
} catch (IOException ex) {
ex.printStackTrace();
} catch (ClassNotFoundException ex) {
ex.printStackTrace();
}
return obj;
}
@Override
public void evict(Object key) {
// TODO Auto-generated method stub
System.out.println("del key");
final String keyf = key.toString();
redisTemplate.execute(new RedisCallback<Long>() {
public Long doInRedis(RedisConnection connection)
throws DataAccessException {
return connection.del(keyf.getBytes());
}
});
}
@Override
public void clear() {
// TODO Auto-generated method stub
System.out.println("clear key");
redisTemplate.execute(new RedisCallback<String>() {
public String doInRedis(RedisConnection connection)
throws DataAccessException {
connection.flushDb();
return "ok";
}
});
}
@Override
public <T> T get(Object key, Class<T> type) {
// TODO Auto-generated method stub
return null;
}
@Override
public <T> T get(Object arg0, Callable<T> arg1) {
// TODO Auto-generated method stub
return null;
}
@Override
public ValueWrapper putIfAbsent(Object arg0, Object arg1) {
// TODO Auto-generated method stub
return null;
}
}
3) 开启springcache注解
4) 在service实现类上添加注解
要在需要缓存的方法或者类上加缓存注解
spring缓存注解@Cacheable
可以标记在一个方法上,也可以标记在一个类上。当标记在一个方法上时表示该方法是支持缓存的,当标记在一个类上时则表示该类所有的方法都是支持缓存的(value key condition)
value属性是必须指定的,其表示当前方法的返回值是会被缓存在哪个Cache上的,对应Cache的名称。其可以是一个Cache也可以是多个Cache,当需要指定多个Cache时其是一个数组。
key属性是用来指定Spring缓存方法的返回结果时对应的key的。该属性支持SpringEL表达式。当我们没有指定该属性时,Spring将使用默认策略生成key。
/**
* 查询所有用户,@Cacheable代表将返回信息放入缓存,缓存名称为"chenjian"的缓存
*/
@Cacheable(value="chenjian")
@Override
public List<Map> findAllUser() {
return userInfoDao.findAllUser();
}
condition有的时候我们可能并不希望缓存一个方法所有的返回结果。通过condition属性可以实现这一功能。condition属性默认为空,表示将缓存所有的调用情形。其值是通过SpringEL表达式来指定的,当为true时表示进行缓存处理;当为false时表示不进行缓存处理,即每次调用该方法时该方法都会执行一次。如下示例表示只有当empno为偶数时才会进行缓存。
/**
* 按照用户id来查询
*TODO key="#id"使用id作为redis缓存key,condition="#id%2==0"代表至缓存id为偶数的数据
*@return
*2018-7-11下午4:28:32
*/
@Cacheable(value="chenjian",key="#id",condition="#id%2==0")
@Override
public UserInfo findUserById(Integer id) {
return userInfoDao.findUserById(id);
}
5) spring缓存注解@CacheEvict
是用来标注在需要清除缓存元素的方法或类上的。当标记在一个类上时表示其中所有的方法的执行都会触发缓存的清除操作
/**
* 根据id删除员工
* CacheEvict能够实时刷新缓存中的数据
*/
@CacheEvict(value="chenjian",key="#id")
@Override
public int deleteUserById(Integer id) {
return userInfoDao.deleteUserById(id);
}
6) spring缓存注解@CachePut
CachePut也可以声明一个方法支持缓存功能。与@Cacheable不同的是使用@CachePut标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,而是每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中。
/**
* 修改员工信息,并同步更新缓存中的员工信息
* @CachePut不管缓存中有没有,没有的话放入缓存,有的话替换缓存
*/
@CachePut(value="chenjian",key="#user.id")
@Override
public UserInfo editUser(UserInfo user) {
userInfoDao.editUser(user);
return user;
}
测试步骤,
首先不加@CachePut(value="chenjian",key="#user.id")
,查询一条数据,这条数据自动缓存到redis,然后修改这条数据,再次查询数据库变化,缓存不变,查询的结果不对。
加上@CachePut(value="chenjian",key="#user.id")
,再次修改同一条数据,查询发现数据库和缓存同步修改。
2. redis(集群版)实现spring缓存
2.1. 修改配置类RedisConfig
@Bean(name = "redisTemplate")
public RedisTemplate getRedisTemplate(){
RedisTemplate template = new StringRedisTemplate(getJedisConnectionFactory());
return template;
}
/**
* 创建redis集群配置信息
*
*/
/*@Bean
public RedisClusterConfiguration getRedisClusterConfiguration() {
RedisClusterConfiguration redisClusterConfiguration = new RedisClusterConfiguration();
redisClusterConfiguration.setMaxRedirects(3);
Set<RedisNode> nodes= new HashSet<RedisNode>();
// 初始化集群节点
for (int i = 7001; i < 7007; i++) {
RedisNode redisNode= new RedisNode(hostName,i);
nodes.add(redisNode);
}
redisClusterConfiguration.setClusterNodes(nodes);
return redisClusterConfiguration;
}*/
/**
* 创建Spring-redis连接池管理工厂 类(bean)(集群版)
*TODO
*@return
*2018-7-11下午1:56:23
*/
/*@Bean
public JedisConnectionFactory getJedisConnectionFactory() {
//构造注入集群信息clusterConfig和redis poolconfig
JedisConnectionFactory jedisConnectionFactory= new JedisConnectionFactory(getRedisClusterConfiguration(),getPoolConfig());
return jedisConnectionFactory;
}*/
本文使用的技术在之前的文章内都有讲过,包括:
redis集群:
建哥手把手系列之30分钟搞定动态添加删除redis集群节点
springboot入门:
SpringBoot快速入门及整合mybatis
springMVC版缓存:
redis之30分钟搞定mybatis缓存(单机+集群)
redis之20分钟轻松搞定springCache缓存(单机+集群)
redis之20分钟轻松搞定springCache缓存(单机+集群)
本文配套有代码和笔记以及高清视频,如有需要请关注我,私信发送关键字“视频”,必回!
相关推荐
- 保持SSH隧道活跃:一个实用的Bash监控脚本
-
引言如果您正在使用AWSDocumentDB或任何位于堡垒主机后面的云托管服务等远程资源,您可能正在使用SSH隧道来安全地访问它们。虽然设置SSH隧道很简单,但保持其活跃状态并监控其状态可能会有些棘...
- 京东大佬问我,为什么说连接池是微服务的关键,你是如何理解的?
-
京东大佬问我,为什么说连接池是微服务的关键,你是如何理解的?我应该如何理解。首先,我需要回忆一下连接池和微服务的基本概念,然后思考它们在微服务架构中的作用和重要性。连接池,数据库连接池,用来管理数据库...
- OOM 血案:5 小时绝地求生,MAT+Arthas 终极排查指南
-
一、血案现场:线上服务突然暴毙2025年4月12日凌晨3点15分,服务突发大规模OOM,三个Pod在10分钟内连续崩溃,Prometheus告警显示JVM堆内存使用率...
- 记Tomcat优化方案
-
Tomcat服务吞吐量评估方案问题:评估方案在一台8核16G的linux服务器上,使用tomcat容器部署服务。在正常情况下如何评估这个tomcat服务可处理的连接数,即服务的吞吐量,请在正常情况下考...
- Java高级面试,常见数据结构的实现原理详细说明及面试总结
-
一、List接口实现类1.ArrayList底层结构:动态数组(Object[]数组)。核心原理:o动态扩容:初始容量为10(JDK1.8),当元素超过容量时,新容量为原容量的1.5倍(old...
- SpringBoot敏感配置项加密与解密实战
-
一、为什么要加密配置?先说说SpringBoot的配置加载机制。我们知道,SpringBoot支持多种配置加载方式,优先级从高到低大概是:命令行参数环境变量application-{profile}....
- 【面试题】nacos 配置管理类型-主配置、共享配置、扩展配置
-
nacos配置管理类型-主配置、共享配置、扩展配置Nacos的配置管理支持多种类型,其中共享配置及其扩展机制(如shared-configs和extension-configs)是微服...
- Spring Boot 的 RedisAutoConfiguration 配置:自动装配到自定义扩展
-
在SpringBoot开发中,Redis作为高性能缓存和分布式数据存储方案被广泛使用。而RedisAutoConfiguration作为SpringBoot自动装配体系的重要组成部分,能...
- Docker图像处理:扩展您的优化工作流程
-
随着应用程序的增长和图像处理需求的增加,传统的优化方法遇到了扩展瓶颈。内存限制、环境不一致和处理瓶颈将图像优化从一个已解决的问题变成了生产环境的噩梦。Docker改变了游戏规则。通过容器化图像处理工作...
- 掌握 Spring 框架这 10 个扩展点,让你的能力更上一层楼
-
当我们提到Spring时,或许首先映入脑海的是IOC(控制反转)和AOP(面向切面编程)。它们可以被视为Spring的基石。正是凭借其出色的设计,Spring才能在众多优秀框架中脱颖而出...
- 简简单单在线文件浏览的功能搞起来很头疼
-
您的系统支持在线预览文件吗?一个小小的问题,背后是无数程序员的爆肝研究,有人说了,我平时打开个文件不是很容易吗?其实不然。文件格式代表着软件行业的底层、高端产出,也代表着经久不衰的使用场景,也是我国底...
- 没硬盘、网盘也能看片自由!NAS一键部署MoonTV,随时随地爽看。
-
本内容来源于@什么值得买APP,观点仅代表作者本人|作者:羊刀仙有没有一个应用服务,能满足既没有足够预算购置硬盘,也不想依托网盘的朋友的家庭观影需求?之前我介绍过LibreTV,本篇再来看看另一个更...
- 阿里云ECS代理商:如何使用ECS部署Node.js应用?
-
Node.js作为一种高性能、事件驱动的JavaScript运行环境,广泛用于构建实时通信、微服务接口、后台管理系统等现代Web应用。而阿里云ECS服务器以高可用性、灵活配置、安全稳定等优势,为部署N...
- 阿里云数据库代理商:如何提高数据库的查询效率?
-
在现代企业应用中,数据库查询效率对整体系统性能的影响巨大。特别是随着数据量的不断增加,如何提升数据库查询的响应速度,成为了数据库优化的关键任务。阿里云提供了一系列工具和策略,帮助用户提升数据库的查询效...
- 阿里云代理商:阿里云G6ne实例如何承载1.4亿QPS?
-
一、阿里云G6ne实例概述1.1G6ne实例的背景与定位阿里云G6ne实例是基于阿里云自主研发的“飞天”架构设计的高性能云服务器实例,专为大规模、需要高IOPS和低延迟的业务场景设计。它采用了更强大的...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- oracle位图索引 (74)
- oracle批量插入数据 (65)
- oracle事务隔离级别 (59)
- oracle主从同步 (56)
- oracle 乐观锁 (53)
- redis 命令 (83)
- php redis (97)
- redis 存储 (67)
- redis 锁 (74)
- 启动 redis (73)
- redis 时间 (60)
- redis 删除 (69)
- redis内存 (64)
- redis并发 (53)
- redis 主从 (71)
- redis同步 (53)
- redis结构 (53)
- redis 订阅 (54)
- redis 登录 (62)
- redis 面试 (58)
- redis问题 (54)
- 阿里 redis (67)
- redis的缓存 (57)
- lua redis (59)
- redis 连接池 (64)