花费4周敲完《黑马点评》的课程,做了详细的笔记,感觉受益匪浅,一直一直都在不停成长着。
突然想起《苍穹外卖》系列至今已收获200+个赞,500+个收藏,好评颇多,私信我的人不计其数,在此谢谢大家。
下一篇开始学习12306订票系统项目,大家敬请期待吧。
(为帮助真正有需要的小伙伴们答疑解惑,减少改BUG的痛苦。源码:3.98r(含简单运行指导)。单独答疑:8.88r / 次。定制化需求(含毕设),面议。面试八股、12306项目,访问网站:www.pbjlovezjy.com,私信咨询。)
首先在数据库连接下新建一个数据库hmdp,然后右键hmdp下的表,选择运行SQL文件,然后指定运行文件hmdp.sql即可(建议MySQL的版本在5.7及以上):
下面这个hm-dianping文件是项目源码。在IDEA中打开。
记得要修改数据库连接和Redis连接的密码:
运行程序后尝试访问:localhost:8081/shop-type/list 进行简单测试:
将nginx文件复制到一个没有中文路径的目录,然后点击nginx.exe运行:
在nginx所在目录打开CMD窗口,输入命令:start nginx.exe
访问:localhost:8080,选择用手机模式看,可以看到具体的页面:
点击发送验证码可以看到验证码发送成功:
controller/UserController中写入如下代码:
@PostMapping("code") public Result sendCode(@RequestParam("phone") String phone, HttpSession session) { //发送短信验证码并保存验证码 return userService.sendCode(phone,session); }
service/IUserService中写入如下代码:
public interface IUserService extends IService { Result sendCode(String phone, HttpSession session); }
service/impl/UserServiceImpl中写入如下代码:
@Service public class UserServiceImpl extends ServiceImpl implements IUserService { @Override public Result sendCode(String phone, HttpSession session) { //校验手机号 if(RegexUtils.isPhoneInvalid(phone)){ //不符合 return Result.fail("手机号格式错误"); } //生成验证码 String code = RandomUtil.randomNumbers(6); //保存验证码到session session.setAttribute("code",code); //发送验证码 log.debug("发送短信验证码成功,验证码:"+code); return Result.ok(); } }
service/impl/UserServiceImpl的UserServiceImpl中写入如下代码:
@Override public Result login(LoginFormDTO loginForm, HttpSession session) { String phone = loginForm.getPhone(); //校验手机 if(RegexUtils.isPhoneInvalid(phone)){ return Result.fail("手机号格式错误"); } //校验验证码 Object cacheCode = session.getAttribute("code"); String code = loginForm.getCode(); if(cacheCode==null || !cacheCode.toString().equals(code)){ //不一致,报错 return Result.fail("验证码错误"); } //一致根据手机号查用户 User user = query().eq("phone", phone).one(); //判断用户是否存在 if(user==null){ //不存在,创建用户并保存 user = createUserWithPhone(loginForm.getPhone()); } //保存用户信息到session session.setAttribute("user",user); return null; } private User createUserWithPhone(String phone){ //1.创建用户 User user = new User(); user.setPhone(phone); user.setNickName(USER_NICK_NAME_PREFIX+RandomUtil.randomString(10)); //2。保存用户 save(user); return user; }
前端点击发送验证码,后端直接把验证码摘抄后输入:
勾选协议然后确定登录,出现如下代码:
然后看到数据库后台记录已更新:
preHandle前置拦截:
postHandle后置拦截:
afterCompletion视图渲染之后返回给用户之前:
在utils下面编写一个LoginInterceptor类,实现preHandle和afterCompletion这两个方法(这里User和UserDto的问题,我推荐的是统一使用UserDto,采用BeanUtils里的copy方法即可):
public class LoginInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { //获取session HttpSession session = request.getSession(); //获取用户 User user = (User) session.getAttribute("user"); //判断用户是否存在 if(user==null){ response.setStatus(401); return false; } UserDTO userDTO = new UserDTO(); BeanUtils.copyProperties(user,userDTO); //存在,保存用户信息的ThreadLocal UserHolder.saveUser(userDTO); //放行 return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { //移除用户 UserHolder.removeUser(); } }
在config下面创建一个MvcConfig类:
通过addInterceptors方法来添加拦截器,registry是拦截器的注册器。
用.excludePathPatterns来排除不需要拦截的路径。在这里code、login、bloghot、shop、shopType、upload和voucher等都不需要拦截。
@Configuration public class MvcConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry){ registry.addInterceptor(new LoginInterceptor()) .excludePathPatterns( "/user/code", "/user/login", "/upload/**", "/blog/hot", "/shop/**", "/shop-type/**", "/voucher/**" ); } }
输入手机号码点击获取验证码,写入返回后端的验证码,勾选协议之后,登录会直接返回首页,此时看我的个人主页没问题:
在P6已将User转为UserDTO返回给前端。
多台Tomcat并不共享session存储空间,当请求切换不同Tomcat服务器时会导致数据丢失的问题。
session的替代方案应该满足:1.数据共享。2.内存存储。3.key、value结构。
想要保存用户的登录信息有2种方法:1.用String类型。2.用Hash类型。
String类型是以JSON字符串格式来保存,比较简单直观,但是占用内存比较多(因为有name和age这类的json格式):
Hash结构可以将对象中的每个字段独立存储,可以针对单个字段做CRUD,并且内存占用更少:
以随机的token作为key来存储用户的数据,token是用一个随机的字符串。
在UserServiceImpl中写入如下代码(调用StringRedisTemplate中的set方法进行数据插入,最好在key的前面加入业务前缀以示区分,形成区分):
@Resource private StringRedisTemplate stringRedisTemplate;
在sendCode这个方法里将保存验证码的代码替换为下面:
//保存验证码到redis stringRedisTemplate.opsForValue().set(LOGIN_CODE_KEY+phone,code,LOGIN_CODE_TTL, TimeUnit.MINUTES);
在login这个方法里进行如下2处修改:
首先是校验验证码:
//校验验证码 String cacheCode = stringRedisTemplate.opsForValue().get(LOGIN_CODE_KEY + phone);
然后是添加把用户信息添加到Redis的逻辑:
//7.保存用户信息到redis---------------- //7.1 随机生成Token作为登录令牌 String token = UUID.randomUUID().toString(true); //7.2 将User对象转为Hash存储 UserDTO userDTO = BeanUtil.copyProperties(user, UserDTO.class); Map userMap = BeanUtil.beanToMap(userDTO); //7.3 存储 stringRedisTemplate.opsForHash().putAll("login:token:"+token,userMap); //7.4设置token有效期 String tokenKey = LOGIN_USER_KEY+token; stringRedisTemplate.expire(tokenKey,LOGIN_USER_TTL,TimeUnit.MINUTES); return Result.ok(token);
在MvcConfig类上有@Configuration注解,说明是由Spring来负责依赖注入。
在MvcConfig类中要编写如下的代码:
@Configuration public class MvcConfig implements WebMvcConfigurer { @Resource private StringRedisTemplate stringRedisTemplate; @Override public void addInterceptors(InterceptorRegistry registry){ registry.addInterceptor(new LoginInterceptor(stringRedisTemplate)) .excludePathPatterns( "/user/code", "/user/login", "/upload/**", "/blog/hot", "/shop/**", "/shop-type/**", "/voucher/**" ); } }
在utils下的LoginInterceptor中写入如下代码:
public class LoginInterceptor implements HandlerInterceptor { @Resource private StringRedisTemplate stringRedisTemplate; public LoginInterceptor(StringRedisTemplate stringRedisTemplate){ this.stringRedisTemplate = stringRedisTemplate; } @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { //TODO;1.获取请求头中的token String token = request.getHeader("authorization"); if(StrUtil.isBlank(token)){ //不存在,拦截,返回401状态码 response.setStatus(401); return false; } //TODO:2.基于TOKEN获取redis的用户 Map
测试:首先把Redis和数据库都启动。 原始的项目的Redis的服务器ID需要更改为自己的。点击发送验证码,redis中有记录,没问题:
但点击登录的时候会报一个无法将Long转String的错误。因为用的是stringRedisTemplate要求所有的字段都是string类型的。
需要对UserServiceImpl中如下的位置进行修改:
Map userMap = BeanUtil.beanToMap(userDTO,new HashMap<>(), CopyOptions.create() .setIgnoreNullValue(true) .setFieldValueEditor((fieldName,fieldValue)->fieldValue.toString()));
效果如下:
现在只有在用户访问拦截器拦截的页面才会刷新页面,假如用户访问的是不需要拦截的页面则不会导致页面的刷新。
现在的解决思路是:新增一个拦截器,拦截一切路径。
复制LoginInterceptor变成一份新的RefreshTokenInterceptor,把下面几处地方改为return true即可:
LoginInterceptor的代码变成如下:
public class LoginInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { //1.判断是否需要拦截(ThreadLocal中是否有用户) if(UserHolder.getUser()==null){ //没有,需要拦截,设置状态码 response.setStatus(401); //拦截 return false; } //放行 return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { //移除用户 UserHolder.removeUser(); } }
现在还需要在MvcConfig里面对拦截器进行更新配置,需要(用order)调整拦截器的执行顺序:
@Configuration public class MvcConfig implements WebMvcConfigurer { @Resource private StringRedisTemplate stringRedisTemplate; @Override public void addInterceptors(InterceptorRegistry registry){ registry.addInterceptor(new LoginInterceptor()) .excludePathPatterns( "/user/code", "/user/login", "/upload/**", "/blog/hot", "/shop/**", "/shop-type/**", "/voucher/**" ).order(1); registry.addInterceptor(new RefreshTokenInterceptor(stringRedisTemplate)) .addPathPatterns("/**").order(0); } }
缓存就是数据交换的缓冲区,是存储数据的临时地方,一般读写性能较高。
缓存作用:降低后端负载;提高读写的效率,降低响应时间。
缓存成本:数据一致性成本(数据库里的数据如果发生变化,容易与缓存中的数据形成不一致)。代码维护成本高(搭建集群)。运营成本高。
在ShopController类的queryShopById方法中:
@GetMapping("/{id}") public Result queryShopById(@PathVariable("id") Long id) { return Result.ok(shopService.queryById(id)); }
在IShopService接口中编写如下代码:
public interface IShopService extends IService { Object queryById(Long id); }
在ShopServiceImpl类的queryById方法中编写具体代码:
@Service public class ShopServiceImpl extends ServiceImpl implements IShopService { @Resource private StringRedisTemplate stringRedisTemplate; @Override public Object queryById(Long id) { String key = CACHE_SHOP_KEY + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isNotBlank(shopJson)){ //3.存在,直接返回 Shop shop = JSONUtil.toBean(shopJson, Shop.class); return Result.ok(shop); } //4.不存在,根据id查询数据库 Shop shop = getById(id); //5.不存在,返回错误 if(shop==null){ return Result.fail("店铺不存在!"); } //6.存在,写入Redis stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop)); return Result.ok(shop); } }
核心是通过调用hutool工具包中的JSONUtil类来实现对象转JSON(方法:toJsonStr(对象))和JSON转对象(方法:toBean(json,Bean的类型))。
TODO:对分类进行缓存。
主动更新:编写业务逻辑,在修改数据库的同时,更新缓存。
适用于高一致性的需求:主动更新,以超时剔除作为兜底方案。
主动更新策略:
1.由缓存的调用者,在更新数据库的同时更新缓存。(一般情况下使用该种方案)
2.缓存与数据库聚合为一个服务,由服务来维护一致性。调用者调用该服务,无需关心缓存的一致性问题。
3.调用者只操作缓存,由其它线程异步的将缓存数据持久化到数据库,保证最终一致。
对1进行分析:
1.选择删除缓存还是更新缓存?如果是更新缓存:每次更新数据库都会更新缓存,无效的写操作比较多。删除缓存:更新数据库时让缓存失效,查询时再更新缓存。
2.如何保证缓存与数据库的操作的同时成功或失败?
单体系统:将缓存与数据库操作放在一个事务。
分布式系统:利用TCC等分布式事务方案。
3.先操作缓存还是先操作数据库?
先删缓存,再操作(写)数据库:
先操作(写)数据库,再删除缓存(出现的概率比较低)
要求线程1来查询的时候缓存恰好失效了->在写入缓存的时候突然来了线程2,对数据库的数据进行了修改->此时线程1写回缓存的是旧数据。
给查询商铺的缓存添加超时剔除和主动更新的策略。
修改ShopController中的业务逻辑,满足下面要求:
1.根据id查询商铺时,如果缓存未命中,则查询数据库,将数据库结果写入缓存,并设置超时时间。
2.根据id修改店铺时,先修改数据库,再删除缓存。
首先修改ShopServiceImpl的redis过期时间:
stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop),CACHE_SHOP_TTL, TimeUnit.MINUTES);
修改ShopController中的updateShop方法:
@PutMapping public Result updateShop(@RequestBody Shop shop) { // 写入数据库 return Result.ok(shopService.update(shop)); }
向IShopService接口中添加update方法:
Object update(Shop shop);
向ShopServiceImpl类中添加update方法:
@Override public Object update(Shop shop) { Long id = shop.getId(); if(id == null){ return Result.fail("商铺id不存在"); } updateById(shop); stringRedisTemplate.delete(CACHE_SHOP_KEY + id); return Result.ok(); }
首先删除缓存中的数据,然后看SQL语句是否执行,是否加上了TTL过期时间。
在PostMan中访问http://localhost:8081/shop,然后修改101茶餐厅为102茶餐厅:
注意要发送的是PUT请求,请求的内容如下:
{ "area": "大关", "openHours": "10:00-22:00", "sold": 4215, "address": "金华路锦昌文华苑29号", "comments": 3035, "avgPrice": 80, "score": 37, "name": "102茶餐厅", "typeId": 1, "id": 1 }
然后去数据库看是否名称更新为102茶餐厅,然后看缓存中的数据是否被删除,用户刷新页面看到102茶餐厅,缓存中会有最新的数据。
缓存穿透指的是客户端请求的数据在缓存中和数据库中都不存在,使得缓存永远不会生效,请求都会打到数据库。
2种解决方法:
1.缓存空对象。优点:实现简单,维护方便。缺点:额外的内存消耗。可能造成短期的不一致(可以设置TTL)。
2.布隆过滤。在客户端和Redis之间加个布隆过滤器(存在不一定存在,不存在一定不存在,有5%的错误率)。
优点:内存占用较少,没有多余key。缺点:实现复杂,存在误判可能。
下图是原始的:
下面是更改后的:
在ShopServiceImpl类里对queryById方法进行修改:
@Override public Object queryById(Long id) { String key = CACHE_SHOP_KEY + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isNotBlank(shopJson)){ //3.存在,直接返回 Shop shop = JSONUtil.toBean(shopJson, Shop.class); return Result.ok(shop); } //上面是有值的情况,下面是无值的2种情况:A:空字符串。B:null。 if(shopJson != null){ return Result.fail("店铺信息不存在!"); } //4.不存在,根据id查询数据库 Shop shop = getById(id); //5.不存在,返回错误 if(shop==null){ stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES); return Result.fail("店铺不存在!"); } //6.存在,写入Redis stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop),CACHE_SHOP_TTL, TimeUnit.MINUTES); return Result.ok(shop); }
测试:
localhost:8080/api/shop/1此时是命中数据。
localhost:8080/api/shop/0此时未命中数据。打开缓存可以看到缓存的是空,并且TTL是200秒。
总结缓存穿透:用户请求的数据在缓存中和数据库中都不存在,不断发起请求,会给数据库造成巨大压力。
缓存穿透:缓存null值和布隆过滤器。还可以增强id的复杂度,避免被猜测id规律。做好数据的基础格式校验。加强用户权限校验。做好热点参数的限流。
缓存雪崩:是指在同一时段大量的缓存key同时失效或者Redis服务宕机,导致大量请求打到数据库,带来巨大的压力。
解决方案:
1.(解决大量缓存key同时失效)给不同Key的TTL添加随机值。
2.(解决Redis宕机)利用Redis集群提高服务的可用性。
3.给缓存业务添加降级限流策略。
4.给业务添加多级缓存(浏览器可以有缓存,nginx可以有缓存,redis可以有缓存,数据库可以有缓存)。
缓存击穿问题:也叫热点key问题,就是一个被高并发访问并且缓存重建业务较复杂的key突然消失了,无数的请求访问在瞬间给数据库带来巨大的冲击。
解决方案:
1.互斥锁。由获取互斥锁成功的线程来查询数据库重建缓存数据。缺点:未获得互斥锁的线程需要等待,性能略差。
2.逻辑过期。设置一个逻辑时间字段,查询缓存的时候检查逻辑时间看是否已过期。如果某个线程获取到互斥锁就开启新线程,由新线程查询数据库重建缓存数据。
其它线程在获取互斥锁失败后不会等待,而是直接返回过期的数据。只有当缓存重建完毕之后释放锁,新线程才会读到最新的数据。
互斥锁优点:
互斥锁没有额外的内存消耗:因为逻辑过期需要维护一个逻辑过期的字段,有额外内存消耗。
互斥锁可以保证强一致性,所有线程拿到的是最新数据。实现也很简单。
互斥锁缺点:
线程需要等待,性能受到影响。可能会有死锁的风险。
逻辑过期优点:
线程无需等待,性能较好。
逻辑过期缺点:
不保证一致性。有额外内存消耗。实现复杂。
在ShopServiceImpl类中定义一个tryLock方法(在Redis中的setnx相当于setIfAbsent方法。)
public boolean tryLock(String key){ Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.SECONDS); return BooleanUtil.isTrue(flag); }
在ShopServiceImpl类中定义一个unLock方法用于解锁。
public void unLock(String key){ stringRedisTemplate.delete(key); }
在ShopServiceImpl类中定义一个queryWithPassThrough方法。
public Shop queryWithPassThrough(Long id){ String key = CACHE_SHOP_KEY + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isNotBlank(shopJson)){ //3.存在,直接返回 Shop shop = JSONUtil.toBean(shopJson, Shop.class); return shop; } //上面是有值的情况,下面是无值的2种情况:A:空字符串。B:null。 if(shopJson != null){ return null; } //4.不存在,根据id查询数据库 Shop shop = getById(id); //5.不存在,返回错误 if(shop==null){ stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES); return null; } //6.存在,写入Redis stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop),CACHE_SHOP_TTL, TimeUnit.MINUTES); return shop; }
在ShopServiceImpl类中定义一个queryWithMutex方法:
public Shop queryWithMutex(Long id){ String key = CACHE_SHOP_KEY + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isNotBlank(shopJson)){ //3.存在,直接返回 Shop shop = JSONUtil.toBean(shopJson, Shop.class); return shop; } //上面是有值的情况,下面是无值的2种情况:A:空字符串。B:null。 if(shopJson != null){ return null; } //4.实现缓存重建 //4.1 获取互斥锁 String lockKey = LOCK_SHOP_KEY+id; Shop shop = null; try { boolean isLock = tryLock(lockKey); //4.2 判断是否获取成功 if(!isLock){ //4.3 失败,则休眠并重试 Thread.sleep(50); return queryWithMutex(id); } //4.4 获取互斥锁成功,根据id查询数据库 shop = getById(id); //模拟重建的延时 Thread.sleep(200); //5.数据库查询失败,返回错误 if(shop==null){ stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES); return null; } //6.存在,写入Redis stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop),CACHE_SHOP_TTL, TimeUnit.MINUTES); } catch (InterruptedException e) { throw new RuntimeException(e); }finally { //7.释放互斥锁 unLock(lockKey); } //8.返回 return shop; }
在ShopServiceImpl类中修改queryById,调用queryWithMutex:
public Object queryById(Long id) { //缓存穿透 //Shop shop = queryWithPassThrough(id); //互斥锁解决缓存击穿 Shop shop = queryWithMutex(id); return Result.ok(shop); }
测试:
定义1000个线程,Ramp-Up时间为5。
请求地址:localhost:8081/shop/1。
设置完毕后点击绿色箭头运行,此时会提示是否保存测试文件,选择不保存(我测试选择保存会报错)。
可以在结果树这里看请求是否发送成功:
先删掉缓存,然后点击绿色箭头发送并发请求,可以发现所有线程请求成功,控制台对数据库的查询只有1次(没有出现多个线程争抢查询数据库的情况),测试成功。
如何添加逻辑过期字段?答:可以在utils包下定义RedisData类(可以让Shop继承RedisData类),也可以在RedisData中设置一个Shop类的data属性:
@Data public class RedisData { private LocalDateTime expireTime; private Object data; }
在ShopServiceImpl类中定义saveShop2Redis方法:
public void saveShop2Redis(Long id,Long expireSeconds){ //1.查询店铺数据 Shop shop = getById(id); //2.封装逻辑过期时间 RedisData redisData = new RedisData(); redisData.setData(shop); redisData.setExpireTime(LocalDateTime.now().plusSeconds(expireSeconds)); //3.写入Redis stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY+id,JSONUtil.toJsonStr(redisData)); }
单元测试,在test包下的HmDianPingApplicationTests中创建testSaveShop类写入测试代码(这里要注意的是输入alt+insert之后选择Test Method要选择Junit 5来进行测试方法的编写):
@SpringBootTest class HmDianPingApplicationTests { @Resource private ShopServiceImpl shopService; @Test void testSaveShop() { shopService.saveShop2Redis(1L,10L); } }
可以看到redis中确实存入了数据:
在ShopServiceImpl中复制一份缓存穿透的代码,更改名称为queryWithLogicalExpire:
private static final ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10); public Shop queryWithLogicalExpire(Long id){ String key = CACHE_SHOP_KEY + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isBlank(shopJson)){ //3.不存在,返回空 return null; } //4.命中,需要先把json反序列化为对象 RedisData redisData = JSONUtil.toBean(shopJson, RedisData.class); JSONObject data = (JSONObject) redisData.getData(); Shop shop = JSONUtil.toBean(data, Shop.class); //5.判断是否过期 //5.1 未过期直接返回店铺信息 LocalDateTime expireTime = redisData.getExpireTime(); if(expireTime.isAfter(LocalDateTime.now())){ return shop; } //5.2 已过期重建缓存 //6.缓存重建 //6.1.获取互斥锁 String lockKey = LOCK_SHOP_KEY + id; boolean isLock = tryLock(lockKey); //6.2.判断是否获取互斥锁成功 if(isLock){ //6.3.成功,开启独立线程,实现缓存重建 CACHE_REBUILD_EXECUTOR.submit(()->{ try { saveShop2Redis(id,20L); //实际中应该设置为30分钟 } catch (Exception e) { throw new RuntimeException(e); } finally { unLock(lockKey); } }); } //6.4.失败,返回过期的商铺信息 return shop; }
测试:
先到数据库把102茶餐厅改为103茶餐厅(因为Redis之前插入了一条缓存为102茶餐厅,并且已经过期,此时数据库与缓存不一致),新的HTTP请求会将逻辑过期的数据删除,然后更新缓存。
线程数设置为100,Ramp-up时间设置为1
在查看结果树里面到中间某个HTTP请求会完成重建,响应数据会改变。
1.安全性问题:在高并发情况下是否会有很多线程来做重建。
2.一致性问题:在重建完成之前得到的是否是旧的数据。
在utils包下创建CacheClient类,先写入如下基础的代码:
@Slf4j @Component public class CacheClient { private final StringRedisTemplate stringRedisTemplate; public CacheClient(StringRedisTemplate stringRedisTemplate) { this.stringRedisTemplate = stringRedisTemplate; } public void set(String key, Object value, Long time, TimeUnit unit){ stringRedisTemplate.opsForValue().set(key,JSONUtil.toJsonStr(value),time,unit); } public void setWithLogicalExpire(String key, Object value,Long expire,TimeUnit unit){ //设置逻辑过期 RedisData redisData = new RedisData(); redisData.setExpireTime(LocalDateTime.now().plusSeconds(unit.toSeconds(expire))); redisData.setData(value); stringRedisTemplate.opsForValue().set(key,JSONUtil.toJsonStr(redisData)); } }
在CacheClient类中编写缓存穿透的共性方法queryWithPassThrough:
public R queryWithPassThrough(String keyPrefix, ID id, Class type, Function dbFallBack,Long time,TimeUnit unit){ String key = keyPrefix + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isNotBlank(shopJson)){ //3.存在,直接返回 return JSONUtil.toBean(shopJson, type); } //上面是有值的情况,下面是无值的2种情况:A:空字符串。B:null。 if(shopJson != null){ return null; } //4.不存在,根据id查询数据库 R r = dbFallBack.apply(id); //5.不存在,返回错误 if(r==null){ stringRedisTemplate.opsForValue().set(key,"",CACHE_NULL_TTL,TimeUnit.MINUTES); return null; } //6.存在,写入Redis this.set(key,r,time,unit); return r; }
编写完queryWithPassThrough之后可以到ShopServiceImpl中直接调用新的方法(记得引入CacheClient类):
@Resource private CacheClient cacheClient; @Override public Object queryById(Long id) { //调用工具类解决缓存击穿 Shop shop = cacheClient.queryWithPassThrough(CACHE_SHOP_KEY, id, Shop.class, this::getById, CACHE_SHOP_TTL, TimeUnit.MINUTES); if(shop==null){ return Result.fail("店铺不存在!"); } return Result.ok(shop); }
进行测试:成功会对不存在的店铺空值进行缓存。
接下来拷贝queryWithLogicalExpire的代码到CacheClient类中进行改写:
private static final ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10); public R queryWithLogicalExpire(String keyPrefix,ID id,Class type,Function dbFallBack,Long time,TimeUnit unit){ String key = keyPrefix + id; //1.从Redis查询缓存 String shopJson = stringRedisTemplate.opsForValue().get(key); //2.判断是否存在 if(StrUtil.isBlank(shopJson)){ //3.不存在,返回空 return null; } //4.命中,需要先把json反序列化为对象 RedisData redisData = JSONUtil.toBean(shopJson, RedisData.class); JSONObject data = (JSONObject) redisData.getData(); R r = JSONUtil.toBean(data, type); //5.判断是否过期 //5.1 未过期直接返回店铺信息 LocalDateTime expireTime = redisData.getExpireTime(); if(expireTime.isAfter(LocalDateTime.now())){ return r; } //5.2 已过期重建缓存 //6.缓存重建 //6.1.获取互斥锁 String lockKey = LOCK_SHOP_KEY + id; boolean isLock = tryLock(lockKey); //6.2.判断是否获取互斥锁成功 if(isLock){ //6.3.成功,开启独立线程,实现缓存重建 CACHE_REBUILD_EXECUTOR.submit(()->{ try { //查询数据库 R r1 = dbFallBack.apply(id); //写入redis this.setWithLogicalExpire(key,r1,time,unit); } catch (Exception e) { throw new RuntimeException(e); } finally { unLock(lockKey); } }); } //6.4.失败,返回过期的商铺信息 return r; } public boolean tryLock(String key){ Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.SECONDS); return BooleanUtil.isTrue(flag); } public void unLock(String key){ stringRedisTemplate.delete(key); }
改写test下的HmDianPingApplicationTests类:
@SpringBootTest class HmDianPingApplicationTests { @Resource private CacheClient cacheClient; @Resource private ShopServiceImpl shopService; @Test void testSaveShop() throws InterruptedException { Shop shop = shopService.getById(1L); cacheClient.setWithLogicalExpire(CACHE_SHOP_KEY+1L,shop,10L,TimeUnit.SECONDS); } }
测试:首先运行HmDianPingApplicationTests类里的测试方法,10秒后逻辑过期,此时运行后台程序,修改数据库1号商铺的name字段,此时访问:localhost:8080/api/shop/1 会出现效果第1次访问为缓存旧值,然后发现缓存过期开始重建,第2次访问开始就是新值。数据库也只有1次重建。
每个店铺都可以发布优惠券,当用户抢购时,就会生成订单并保存到tb_voucher_order这张表中,而订单表如果使用数据库自增ID会存在一些问题。
1.id的规律性太明显。
2.受单表数据量的限制(分表之后每张表都自增长,id会出现重复)。
全局ID生成器:是一种在分布式系统下用来生成全局唯一ID的工具。
要求全局唯一ID生成器满足如下几点:1.唯一性。2.高可用。3.高性能。4.递增性。5.安全性。
为了增加ID的安全性,我们可以不直接使用Redis自增的数值,而是拼接一些其它信息。
符号位永远为0代表整数。
31位的时间戳是以秒为单位,定义了一个起始时间,用当前时间减起始时间,预估可以使用69年。
32位的是序列号是Redis自增的值,支持每秒产生2^32个不同ID。
在utils包下定义一个RedisWorker类,是一个基于Redis的ID生成器。
如果只使用一个key来自增记录有一个坏处,最终key的自增数量会突破容量的上限,假如自增超过32位彼时便无法再存储新的数据,解决的方案是采用拼接日期。
@Component public class RedisIdWorker { private static final long BEGIN_TIMESTAMP = 1640995200L; //序列号的位数 private static final int COUNT_BITS=32; private StringRedisTemplate stringRedisTemplate; public RedisIdWorker(StringRedisTemplate stringRedisTemplate) { this.stringRedisTemplate = stringRedisTemplate; } public long nextId(String keyPrefix){ //1.生成时间戳 LocalDateTime now = LocalDateTime.now(); long timeStamp = now.toEpochSecond(ZoneOffset.UTC) - BEGIN_TIMESTAMP; //2.生成序列号 //2.1获取当前日期,精确到天 String date = now.format(DateTimeFormatter.ofPattern("yyyy:MM:dd")); //2.2自增长 long count = stringRedisTemplate.opsForValue().increment("icr:" + keyPrefix + ":" + date); //3.拼接并返回 return timeStamp << COUNT_BITS | count; } }
在HmDianPingApplicationTests中写入如下的测试代码:
@Resource private ShopServiceImpl shopService; @Resource private RedisIdWorker redisIdWorker; private ExecutorService es = Executors.newFixedThreadPool(500); @Test void testIdWorker() throws InterruptedException { CountDownLatch latch = new CountDownLatch(300); Runnable task = ()->{ for(int i=0;i<100;i++){ long id = redisIdWorker.nextId("order"); System.out.println("id="+id); } latch.countDown(); }; long begin = System.currentTimeMillis(); for(int i=0;i<300;i++){ es.submit(task); } latch.await(); long end = System.currentTimeMillis(); System.out.println("Result Time = " + (end-begin)); }
运行之后可以看到以十进制输出的所有编号:
可以在Redis中看到自增长的结果,1次是30000:
大概2秒可以生成3万条,速度还是可以的。
全局唯一ID生成策略:
1.UUID利用JDK自带的工具类即可生成,生成的是16进制的字符串,无单调递增的特性。
2.Redis自增(每天一个key,方便统计订单量。时间戳+计数器的格式。)
3.snowflake雪花算法(不依赖于Redis,性能更好,对于时钟依赖)
4.数据库自增
每个店铺都可以发放优惠券,分为平价券和特价券。平价券可以任意抢购,特价券需要秒杀抢购。
tb_voucher:优惠券基本信息,优惠金额,使用规则等。
tb_seckill_voucher:优惠券的库存,开始抢购时间,结束抢购时间,只有特价优惠券才需要填写这些信息。
请求的信息如下可自行复制(注意beginTime和endTime需要修改):
{ "shopId":1, "title":"100元代金券", "subTitle":"周一至周五均可使用", "rules":"全场通用\\n无需预约\\n可无限叠加\\不兑现、不找零\\n仅限堂食", "payValue":8000, "actualValue":10000, "type":1, "stock":100, "beginTime":"2024-04-10T10:09:17", "endTime":"2024-04-11T12:09:04" }
注意要在请求头中带Authorization参数否则会报401(登录后进入“我的”页面,看网络包有Authorization的值):
以如下格式发送请求:
首先在tb_voucher表中可以看到新增的优惠券:
在tb_seckill_voucher表中也可以看到秒杀优惠券的具体信息:
在前端也能看到新增的100元代金券,注意优惠券的时间一定要进行更改,如果不在开始和结束时间区间内优惠券会处于下架状态是看不到的。
首先要判断秒杀是否开始或结束,所以要先查询优惠券的信息,如果尚未开始或者已经结束无法下单。
要判断库存是否充足,如果不足则无法下单。
在VouchrOrderController类中:
@RestController @RequestMapping("/voucher-order") public class VoucherOrderController { @Resource private IVoucherService voucherService; @PostMapping("seckill/{id}") public Result seckillVoucher(@PathVariable("id") Long voucherId) { return voucherService.seckillVoucher(voucherId); } }
在IVoucherOrderService中写入如下代码:
public interface IVoucherOrderService extends IService { Result seckillVoucher(Long voucherId); }
在VoucherOrderServiceImpl中写入如下代码:
@Service @Transactional public class VoucherOrderServiceImpl extends ServiceImpl implements IVoucherOrderService { @Resource private ISeckillVoucherService seckillVoucherService; @Resource private RedisIdWorker redisIdWorker; @Override public Result seckillVoucher(Long voucherId) { //1.查询优惠券信息 SeckillVoucher voucher = seckillVoucherService.getById(voucherId); //2.判断秒杀是否开始 //2.1秒杀尚未开始返回异常 if(voucher.getBeginTime().isAfter(LocalDateTime.now())){ return Result.fail("秒杀尚未开始"); } //2.2秒杀已结束返回异常 if(voucher.getEndTime().isBefore(LocalDateTime.now())){ return Result.fail("秒杀已经结束"); } //3.判断库存是否充足 if(voucher.getStock()<1){ //3.1库存不足返回异常 return Result.fail("库存不足!"); } //3.2库存充足扣减库存 boolean success = seckillVoucherService.update() .setSql("stock = stock - 1") .eq("voucher_id", voucherId).update(); if(!success){ return Result.fail("库存不足!"); } //4.创建订单,返回订单id VoucherOrder voucherOrder = new VoucherOrder(); long orderId = redisIdWorker.nextId("order");//订单id voucherOrder.setId(orderId); Long userId = UserHolder.getUser().getId();//用户id voucherOrder.setUserId(userId); voucherOrder.setVoucherId(voucherId);//代金券id save(voucherOrder); return Result.ok(orderId); } }
测试:点击限时抢购之后会提示抢购成功。
Jmeter的配置如下:
注意Authorization要事先登录获取:
下面是结果:
发现tb_seckill_voucher中库存为-9,在tb_voucher_order中插入了109条数据,说明出现了超卖的问题。
正常逻辑:
非正常逻辑:
超卖问题是典型的多线程安全问题,针对这一问题的常见解决方案是加锁。
悲观锁:认为线程安全问题一定会发送,因此在操作数据之前要先获取锁,确保线程串行执行。像Synchronized、Lock都属于悲观锁。
乐观锁:认为线程安全问题不一定会发生,因此不加锁,只是在更新数据时去判断有没有其它线程对数据做了修改。
如果没有修改则认为是安全的,自己才更新数据。
如果已经被其它线程修改说明发生了安全问题,此时可以重试或异常。‘
乐观锁关键是判断之前查询得到的数据是否被修改过,常见的方法有2种:
1.版本号法:
2.CAS法(版本号法的简化版):查询的时候把库存查出来,更新的时候判断库存和之前查到的库存是否一致,如果一致则更新数据。
只需加上下面这段代码即可:.eq("stock",voucher.getStock()) 。用于比较当前数据库的库存值和之前查询到的库存值是否相同,只有相同时才可以执行set语句。
//3.2库存充足扣减库存 boolean success = seckillVoucherService.update() .setSql("stock = stock - 1") //相当于set条件 set stock = stock - 1 .eq("voucher_id", voucherId) //相当于where条件 where id = ? and stock = ? .eq("stock",voucher.getStock()).update();
但现在出现了异常值偏高的问题,正常的请求大约只占10%。
原理是因为:假如一次有30个线程涌入,查询到库存值为100,只有1个线程能把值改为99,其它29个线程比对库存值99发现和自己查询到的库存值100不同,所以都认为数据已经被修改过,所以都失败了。
乐观锁的问题,成功率太低。
现在只需要保证stock>0即可,只要存量大于0就可以任意扣减。
boolean success = seckillVoucherService.update() .setSql("stock = stock - 1") //相当于set条件 set stock = stock - 1 .eq("voucher_id", voucherId) //相当于where条件 where id = ? and stock = ? .gt("stock",0).update();
乐观锁缺陷:
需要大量对数据库进行访问,容易导致数据库的崩溃。
总结:
修改秒杀业务,要求同一个优惠券,一个用户只能下一单。
首先不建议把锁加在方法上,因为任何一个用户来了都要加这把锁,而且是同一把锁,方法之间变成串行执行,性能很差。
因此可以把锁加在用户id上,只有当id相同时才会对锁形成竞争关系。但是因为toString的内部是new了一个String字符串,每调一次toString都是生成一个全新的字符串对象,锁对象会变。
所以可以调用intern()方法,intern()方法会优先去字符串常量池里查找与目标字符串值相同的引用返回(只要字符串一样能保证返回的结果一样)。
但是因为事务是在函数执行结束之后由Spring进行提交,如果把锁加在createVoucherOrder内部其实有点小——因为如果解锁之后,其它线程可以进入,而此时事务尚未提交,仍然会导致安全性问题。
因此最终方案是把synchronized加在createVoucherOrder的方法外部,锁住的是用户id。
关于代理对象事务的问题:通常情况下,当一个使用了@Transactional注解的方法被调用时,Spring会从上下文中获取一个代理对象来管理事务。
但是如果加@Transactional方法是被同一个类中的另一个方法调用时,Spring不会使用代理对象,而是直接调用该方法,导致事务注解失效。
为避免这种情况,可以使用AopContext.currentProxy方法获取当前的代理对象,然后通过代理对象调用被@Transactional注解修饰的方法,确保事务生效。
在VoucherOrderServiceImpl中写入如下代码(注意:ctrl+alt+m可以把含有return的代码段进行提取):
@Service public class VoucherOrderServiceImpl extends ServiceImpl implements IVoucherOrderService { @Resource private ISeckillVoucherService seckillVoucherService; @Resource private RedisIdWorker redisIdWorker; @Override public Result seckillVoucher(Long voucherId) { //1.查询优惠券信息 SeckillVoucher voucher = seckillVoucherService.getById(voucherId); //2.判断秒杀是否开始 //2.1秒杀尚未开始返回异常 if(voucher.getBeginTime().isAfter(LocalDateTime.now())){ return Result.fail("秒杀尚未开始"); } //2.2秒杀已结束返回异常 if(voucher.getEndTime().isBefore(LocalDateTime.now())){ return Result.fail("秒杀已经结束"); } voucher = seckillVoucherService.getById(voucherId); //3.判断库存是否充足 if(voucher.getStock()<1){ //3.1库存不足返回异常 return Result.fail("库存不足!"); } Long userId = UserHolder.getUser().getId(); synchronized (userId.toString().intern()){ //获取代理对象 IVoucherOrderService proxy = (IVoucherOrderService) AopContext.currentProxy(); return proxy.createVoucherOrder(voucherId); } } @Transactional public Result createVoucherOrder(Long voucherId) { //6.一人一单 Long userId = UserHolder.getUser().getId(); //6.1查询订单 int count = query().eq("user_id", userId).eq("voucher_id", voucherId).count(); //6.2判断是否存在 if(count>0){ //用户已经购买过了 return Result.fail("用户已经购买过一次!"); } //3.2库存充足扣减库存 boolean success = seckillVoucherService.update() .setSql("stock = stock - 1") //相当于set条件 set stock = stock - 1 .eq("voucher_id", voucherId) //相当于where条件 where id = ? and stock = ? .gt("stock",0).update(); if(!success){ return Result.fail("库存不足!"); } //4.创建订单,返回订单id VoucherOrder voucherOrder = new VoucherOrder(); long orderId = redisIdWorker.nextId("order");//订单id voucherOrder.setId(orderId); voucherOrder.setUserId(userId); voucherOrder.setVoucherId(voucherId);//代金券id save(voucherOrder); return Result.ok(orderId); } }
在IVoucherOrderService接口中加入下面这个方法:
Result createVoucherOrder(Long voucherId);
在pom.xml中引入如下的依赖:
org.aspectj aspectjweaver
在启动类HmDianPingApplication上加如下注解:
@EnableAspectJAutoProxy(exposeProxy = true)
测试: 成功实现一名用户只能领取一张优惠券。
本P主要是为了验证在集群下synchronized并不能保证线程的并发安全。
如下图可以设置项目启动的端口号,确保启动的项目之间端口号不同:
在nginx.conf中放开8082的这个配置:
向下面这个页面发送请求:
http://localhost:8080/api/voucher/list/1
可以看到请求会分别被8082和8081接收,是轮询的效果:
首先到tb_voucher_order把之前的订单删除,到tb_seckill_voucher中把stock重新改回100。
准备2个相同的秒杀请求:要注意请求的地址是:http://localhost:8080/api/voucher-order/seckill/13
我这里直接用Jemeter来进行测试,模拟高并发场景:
下面是效果:可以看到并发请求能够同时进入集群的每台结点。
正常情况:
在集群模式下,每一个节点都是一个全新的JVM,每个JVM都有自己的锁。锁监视器只能在当前JVM的范围内,监视线程实现互斥。
现在就要实现让多个JVM使用的是同一把锁。跨JVM、跨进程的锁。
synchronized只能保证单个JVM内部的多个线程之间的互斥,而没法让集群下多个JVM进程间的线程互斥。
要让多个JVM进程能看到同一个锁监视器,而且同一时间只有一个线程能拿到锁监视器。
所以必须使用分布式锁,分布式锁:满足分布式系统或集群模式下多进程可见并且互斥的锁。
分布式锁要满足:多进程可见+互斥+高可用+高性能+安全性。
分布式锁可以通过MySQL或Redis或Zookeeper来实现。
MySQL:
1.互斥:是利用mysql本身的互斥锁机制。在执行写操作的时候,MySQL会自动分配一个互斥的锁。
2.可用性:好。3.性能:受限于MySQL性能。
4.安全性:事务机制,如果断开连接,会自动释放锁。
Redis:
1.互斥:利用setnx这样的互斥命令。往Redis里set数据只有不存在时才能set成功。
2.可用性:好,Redis支持主从和集群。3.性能:好。
4.安全性:如果没有执行删除key的操作,key不会自动释放。但可以利用锁的超时机制,到期自动释放。
Zookeeper:
1.利用节点的唯一性(节点不重复)和有序性(节点递增)实现互斥。利用有序性:id最小的节点获取锁成功;释放锁只需要删除id最小的节点。
2.可用性:好。3.性能:比Redis差,一般,强调强一致性,主从间同步需要时间。
4.安全性:好。因为是临时节点,断开连接会自动释放。
假如获取锁后宕机,锁无法释放——>可以添加超时过期时间。
为了防止锁在SETEX和EXPIRE之间过期,可以直接用一条命令(原子操作)来实现设置过期时间(EX)和只有lock不存在时才能设置(NX)。
采用非阻塞式获取锁,如果成功返回true,失败返回false。
在utils下面创建一个ILock接口:
public interface ILock { //尝试获取锁 boolean tryLock(long timeoutSec); //释放锁 void unlock(); }
在utils下面实现SimpleRedisLock类:
public class SimpleRedisLock implements ILock { private String name; private StringRedisTemplate stringRedisTemplate; public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) { this.name = name; this.stringRedisTemplate = stringRedisTemplate; } private static final String KEY_PREFIX = "lock:"; @Override public boolean tryLock(long timeoutSec) { //获取线程标示 long threadId = Thread.currentThread().getId(); Boolean success = stringRedisTemplate.opsForValue().setIfAbsent(KEY_PREFIX+name,threadId+"",timeoutSec, TimeUnit.SECONDS); return Boolean.TRUE.equals(success); } @Override public void unlock() { //释放锁 stringRedisTemplate.delete(KEY_PREFIX+name); } }
更改VoucherOrderServiceImpl类中的seckillVoucher方法的代码:
@Resource private RedisIdWorker redisIdWorker; @Resource private StringRedisTemplate stringRedisTemplate; @Override public Result seckillVoucher(Long voucherId) { //1.查询优惠券信息 SeckillVoucher voucher = seckillVoucherService.getById(voucherId); //2.判断秒杀是否开始 //2.1秒杀尚未开始返回异常 if(voucher.getBeginTime().isAfter(LocalDateTime.now())){ return Result.fail("秒杀尚未开始"); } //2.2秒杀已结束返回异常 if(voucher.getEndTime().isBefore(LocalDateTime.now())){ return Result.fail("秒杀已经结束"); } voucher = seckillVoucherService.getById(voucherId); //3.判断库存是否充足 if(voucher.getStock()<1){ //3.1库存不足返回异常 return Result.fail("库存不足!"); } Long userId = UserHolder.getUser().getId(); SimpleRedisLock lock = new SimpleRedisLock("order:"+userId,stringRedisTemplate); boolean isLock = lock.tryLock(1200); //判断是否获取锁成功 if(!isLock) { return Result.fail("不允许重复下单"); } try { //获取代理对象 IVoucherOrderService proxy = (IVoucherOrderService) AopContext.currentProxy(); return proxy.createVoucherOrder(voucherId); }finally { lock.unlock(); } }
经测试多台节点相同用户只能获取同一张优惠券成功:
假如某个线程(线程A)获取到锁之后,出现了业务阻塞,导致阻塞时间超过了锁自动释放的时间,锁因超时自动释放。此时其它线程(线程B)过来拿到了锁,开始执行业务。但线程A此时业务执行完毕,释放了锁,但释放的是线程B的锁。此时线程C过来看锁已被释放,趁虚而入拿到锁,此时线程B和线程C是并行执行。
要解决这个问题:线程在删除锁之前要先看锁是否是自己加的(获取锁的标示并判断是否一致)。
1.在获取锁时存入线程标示(可以用UUID表示)。
2.在释放锁时先获取锁中的线程标示,判断是否与当前线程标示一致(如果一致释放锁,如果不一致则不释放锁)。
首先要修改SimpleRedisLock里面的如下代码,主要是调用hutool工具包生成UUID(每次线程调用都会生成一个唯一的UUID),让Redis的前缀变成UUID+线程ID:
private static final String ID_PREFIX = UUID.fastUUID().toString(true)+"-"; @Override public boolean tryLock(long timeoutSec) { //获取线程标示 String threadId = ID_PREFIX + Thread.currentThread().getId(); Boolean success = stringRedisTemplate.opsForValue().setIfAbsent(KEY_PREFIX+name,threadId,timeoutSec, TimeUnit.SECONDS); return Boolean.TRUE.equals(success); }
现在要修改的是SimpleRedisLock类里面的unlock方法,主要是比较当前线程的标示和Redis中锁的标示是否一致,只有标示一致才能释放锁:
@Override public void unlock() { //获取线程标示 String threadId = ID_PREFIX + Thread.currentThread().getId(); //获取锁中的标示 String id = stringRedisTemplate.opsForValue().get(KEY_PREFIX + name); if(threadId.equals(id)){ //释放锁 stringRedisTemplate.delete(KEY_PREFIX+name); } }
现在假设出现了其它问题,比如线程1在判断完锁标示是否一致之后出现了阻塞(比如JVM垃圾回收FULL GC导致阻塞了过长时间),此时锁超时了,线程2趁虚而入获取了锁,此时线程1直接释放了线程2的锁,此时线程3趁虚而入继续给Redis加锁,此时会出现线程2和线程3并行执行。
根本的原因是:获取锁标示和释放锁的操作不是原子性的,现在要解决的问题就是将这两个操作变成原子性的。
Redis提供Lua脚本功能,在一个脚本中编写多条Redis命令,确保多条命令执行时的原子性。
Lua是一种编程语言,它的基本语法可以参考网站:https://www.runoob.com/lua/lua-tutorial.html
执行脚本的方法:
执行一个写死的set命令:
在Lua语言里,数组的第一个元素下标是1。
繁琐版的Lua脚本内容如下:
-- 锁的key local key = KEYS[1] -- 当前线程标示 local threadId = ARGV[1] --获取锁中的线程标示 local id = redis.call('get',key) --比较线程标示与锁中的标示是否一致 if(id == threadId) then --释放锁 del key return redis.call('del',key) end return 0
简化版的Lua脚本内容如下:
--比较线程标示与锁中的标示是否一致 if(redis.call('get',KEYS[1]) == ARGV[1]) then --释放锁 del key return redis.call('del',KEYS[1]) end return 0
在resources下创建unlock.lua,会提示下载一个plugins点击install,然后只需要下载一个EmmyLua即可,实测如果下载了多个Lua相关的插件会产生冲突,最终导致IDEA打不开,这真是血泪的教训!
在SimpleRedisLock中写入如下的代码,因为我们希望的是在一开始就将Lua的脚本加载好,而不是等到要调用释放锁的时候再去加载Lua脚本,所以采用静态变量和静态代码块,这些部分在类初始化的时候就会被加载:
private static final DefaultRedisScript UNLOCK_SCRIPT; static { UNLOCK_SCRIPT = new DefaultRedisScript<>(); UNLOCK_SCRIPT.setLocation(new ClassPathResource("unlock.lua")); UNLOCK_SCRIPT.setResultType(Long.class); }
在SimpleRedisLock类的unlock方法中写入如下的代码:
@Override public void unlock() { stringRedisTemplate.execute(UNLOCK_SCRIPT, Collections.singletonList(KEY_PREFIX+name), ID_PREFIX + Thread.currentThread().getId()); }
在程序1和程序2的下面这个位置打上断点:
在测试API中测试访问如下的URL:
http://localhost:8080/api/voucher-order/seckill/14
分别测试秒杀优惠券1和2:
在Redis中能看到程序1获取锁成功,然后直接把lock锁删掉,模拟超时释放的情况:
然后让程序2往下走一步,可以看到程序2获取到了锁
然后可以直接放行程序1,会看到结果是程序2加的锁没有被删除。
最后放行程序2,会看到程序2加的锁被删除。
总结:
基于Redis的分布式锁的实现思路:
1.利用set nx ex获取锁,并设置过期时间,保存线程标示。
2.释放锁时先判断线程标示是否与自己一致,一致则删除锁。
特性:
1.利用set nx满足互斥性。
2.利用set nx保障故障时锁依然能够释放,避免死锁,提高安全性。
3.利用Redis集群保障高可用和高并发的特性。
目前基于setnx实现的分布式锁存在以下几个问题:
1.不可重入:同一线程无法多次获取同一把锁。
2.不可重试:获取锁只尝试一次就返回false,没有重试机制。
3.超时释放:锁超时释放虽然可以避免死锁,但如果是业务执行耗时较长,也会导致锁释放存在安全隐患。
4.主从一致性:如果Redis提供了主从集群,主从同步存在延迟,当主节点宕机时,如果从节点还未同步主节点中的锁数据,则会出现锁信息的不一致。
Redisson是一个在Redis的基础上实现的Java驻内存数据网格。它不仅提供了一系列的分布式的Java常用对象,还提供了许多分布式服务,其中包含了各种分布式锁的实现。
第1步,先引入依赖:
org.redisson redisson 3.13.6
第2步,在config包下创建RedissonConfig类,写入如下代码:
@Configuration public class RedissonConfig{ @Bean public RedissonClient redissonClient(){ //配置 Config config = new Config(); config.useSingleServer().setAddress("redis://127.0.0.1:6379").setPassword(""); //创建RedissonClient对象 return Redisson.create(config); } }
第3步,引入RedissonClient,调用getLock获取锁对象,然后用tryLock获取锁。
第4步,启动服务
发送下面的请求:
在执行释放锁的语句前,可以看到Redis中有锁的记录:
用jmeter来测试,可以发现没有出现并发安全问题:
ReentrantLock可重入锁的原理:获取锁的时候在判断这个锁已经被占有的情况下,会检查占有锁的是否是当前线程,如果是当前线程,也会获取锁成功。会有一个计数器记录重入的次数。
会通过下面的结构来记录某个线程重入了几次锁。
每释放一次锁采用的策略是把重入次数减1。
加锁和释放锁是成对出现的,因此当方法执行到最外层结束时,重入的次数一定会减为0。
1.是否存在锁
2.存在锁,判断是否是自己的。
是,锁计数+1。
不是,获取锁失败。
3.不存在锁
获取锁,添加线程标示。
Redisson底层可重入锁加锁的逻辑:
Redisson底层可重入锁解锁的逻辑:
下面是对含有waitTime(等待时间)的tryLock的跟踪:
看门狗超时时间是30秒
subscribeFuture.await等待的是释放锁的通知,如果future在指定时间内获得,返回true,等待的是time的时间,time是锁的剩余最大等待时间。
如果超时返回false,然后会进到cancel里,调用unsubscribe方法,取消订阅。
不是无休止的忙等机制,而是只有当锁释放后获得通知后才进行加锁尝试,在没收到通知前是被阻塞状态。
下面是定时更新锁的有效期的逻辑:
相当于设置了一个定时任务每隔10秒重置一次有效期。
定时任务的结束是在解锁的逻辑当中:
获取锁机制:
1.判断ttl是否为null
1.1 为null,获取锁成功(涉及自动更新锁过期时间),判断leaseTime是否为-1
1.1.1 为-1自动开启看门狗机制,定时更新锁的过期时间
看门狗默认30秒,每隔10秒会更新一次过期时间。
1.1.2 不为-1返回true
1.2 不为null,获取锁失败(涉及获取锁的失败重试),判断剩余等待时间是否大于0
1.2.1 大于0,订阅并等待释放锁的信号
在受到释放信号后会判断是否超时,如未超时继续尝试获取锁
1.2.2 不大于0,获取锁失败
释放锁机制:
1.尝试释放锁,判断是否成功
1.1 释放成功。
发送锁释放的消息(与获取锁的失败重试关联)
取消看门狗机制(与自动更新锁过期时间关联)
1.2 释放失败。返回异常。
Redisson是如何解决可重入问题、获取锁的失败重试、锁超时释放问题的?
可重入问题:利用哈希表记录线程id和重入次数。
获取锁的失败重试:利用消息订阅和信号量方式实现获取锁失败时的等待、唤醒和锁的重试获取。
锁超时释放:利用看门狗机制,每隔一段时间,重置超时时间。
主节点负责写,从节点负责读,主节点和从节点间需要同步,会存在延迟。
如果主节点宕机,会从从节点中选拔一个新的节点作为主节点。如果主从同步尚未完成,会出现锁失效的问题。
现在在所有主节点中都存放一份锁,要求一个线程必须从所有主节点中获取锁,才算真正获取锁。
假如此时有一个主节点宕机,恰好主从同步没有完成,此时有其它线程趁虚而入获取到了新主节点的锁,但因为没能获取其它主节点的锁,因此也是获取锁失败的。
这种锁叫作联锁。
秒杀业务流程:
1.扣减优惠券的库存(不能超卖,判断库存是否充足)
2.将用户抢购的优惠券信息写入订单,完成订单的创建
3.一个用户对一个优惠券只能下一单
为了获取1000名用户的token,我爆肝1h写了下面的生成代码。
生成效果如下:共计1008名用户,给每位用户生成了专属的token:
并且把所有token存入了output.txt文件中,方便Jmeter读取:
下面是生成代码,把代码放入UserServiceImpl中,然后在login方法中调用即可:
public void generateToken() { String[] phoneNumbers = { "13456762069", "13456789001", "13456789011", "13686869696", "13688668889", "13688668890", "13688668891", "13688668892", "13688668893", "13688668894", "13688668895", "13688668896", "13688668897", "13688668898", "13688668899", "13688668900", "13688668901", "13688668902", "13688668903", "13688668904", "13688668905", "13688668906", "13688668907", "13688668908", "13688668909", "13688668910", "13688668911", "13688668912", "13688668913", "13688668914", "13688668915", "13688668916", "13688668917", "13688668918", "13688668919", "13688668920", "13688668921", "13688668922", "13688668923", "13688668924", "13688668925", "13688668926", "13688668927", "13688668928", "13688668929", "13688668930", "13688668931", "13688668932", "13688668933", "13688668934", "13688668935", "13688668936", "13688668937", "13688668938", "13688668939", "13688668940", "13688668941", "13688668942", "13688668943", "13688668944", "13688668945", "13688668946", "13688668947", "13688668948", "13688668949", "13688668950", "13688668951", "13688668952", "13688668953", "13688668954", "13688668955", "13688668956", "13688668957", "13688668958", "13688668959", "13688668960", "13688668961", "13688668962", "13688668963", "13688668964", "13688668965", "13688668966", "13688668967", "13688668968", "13688668969", "13688668970", "13688668971", "13688668972", "13688668973", "13688668974", "13688668975", "13688668976", "13688668977", "13688668978", "13688668979", "13688668980", "13688668981", "13688668982", "13688668983", "13688668984", "13688668985", "13688668986", "13688668987", "13688668988", "13688668989", "13688668990", "13688668991", "13688668992", "13688668993", "13688668994", "13688668995", "13688668996", "13688668997", "13688668998", "13688668999", "13688669000", "13688669001", "13688669002", "13688669003", "13688669004", "13688669005", "13688669006", "13688669007", "13688669008", "13688669009", "13688669010", "13688669011", "13688669012", "13688669013", "13688669014", "13688669015", "13688669016", "13688669017", "13688669018", "13688669019", "13688669020", "13688669021", "13688669022", "13688669023", "13688669024", "13688669025", "13688669026", "13688669027", "13688669028", "13688669029", "13688669030", "13688669031", "13688669032", "13688669033", "13688669034", "13688669035", "13688669036", "13688669037", "13688669038", "13688669039", "13688669040", "13688669041", "13688669042", "13688669043", "13688669044", "13688669045", "13688669046", "13688669047", "13688669048", "13688669049", "13688669050", "13688669051", "13688669052", "13688669053", "13688669054", "13688669055", "13688669056", "13688669057", "13688669058", "13688669059", "13688669060", "13688669061", "13688669062", "13688669063", "13688669064", "13688669065", "13688669066", "13688669067", "13688669068", "13688669069", "13688669070", "13688669071", "13688669072", "13688669073", "13688669074", "13688669075", "13688669076", "13688669077", "13688669078", "13688669079", "13688669080", "13688669081", "13688669082", "13688669083", "13688669084", "13688669085", "13688669086", "13688669087", "13688669088", "13688669089", "13688669090", "13688669091", "13688669092", "13688669093", "13688669094", "13688669095", "13688669096", "13688669097", "13688669098", "13688669099", "13688669100", "13688669101", "13688669102", "13688669103", "13688669104", "13688669105", "13688669106", "13688669107", "13688669108", "13688669109", "13688669110", "13688669111", "13688669112", "13688669113", "13688669114", "13688669115", "13688669116", "13688669117", "13688669118", "13688669119", "13688669120", "13688669121", "13688669122", "13688669123", "13688669124", "13688669125", "13688669126", "13688669127", "13688669128", "13688669129", "13688669130", "13688669131", "13688669132", "13688669133", "13688669134", "13688669135", "13688669136", "13688669137", "13688669138", "13688669139", "13688669140", "13688669141", "13688669142", "13688669143", "13688669144", "13688669145", "13688669146", "13688669147", "13688669148", "13688669149", "13688669150", "13688669151", "13688669152", "13688669153", "13688669154", "13688669155", "13688669156", "13688669157", "13688669158", "13688669159", "13688669160", "13688669161", "13688669162", "13688669163", "13688669164", "13688669165", "13688669166", "13688669167", "13688669168", "13688669169", "13688669170", "13688669171", "13688669172", "13688669173", "13688669174", "13688669175", "13688669176", "13688669177", "13688669178", "13688669179", "13688669180", "13688669181", "13688669182", "13688669183", "13688669184", "13688669185", "13688669186", "13688669187", "13688669188", "13688669189", "13688669190", "13688669191", "13688669192", "13688669193", "13688669194", "13688669195", "13688669196", "13688669197", "13688669198", "13688669199", "13688669200", "13688669201", "13688669202", "13688669203","13688669204", "13688669205", "13688669206", "13688669207", "13688669208", "13688669209", "13688669210", "13688669211", "13688669212", "13688669213", "13688669214", "13688669215", "13688669216", "13688669217", "13688669218", "13688669219", "13688669220", "13688669221", "13688669222", "13688669223", "13688669224", "13688669225", "13688669226", "13688669227", "13688669228", "13688669229", "13688669230", "13688669231", "13688669232", "13688669233", "13688669234", "13688669235", "13688669236", "13688669237", "13688669238", "13688669239", "13688669240", "13688669241", "13688669242", "13688669243", "13688669244", "13688669245", "13688669246", "13688669247", "13688669248", "13688669249", "13688669250", "13688669251", "13688669252", "13688669253", "13688669254", "13688669255", "13688669256", "13688669257", "13688669258", "13688669259", "13688669260", "13688669261", "13688669262", "13688669263", "13688669264", "13688669265", "13688669266", "13688669267", "13688669268", "13688669269", "13688669270", "13688669271", "13688669272", "13688669273", "13688669274", "13688669275", "13688669276", "13688669277", "13688669278", "13688669279", "13688669280", "13688669281", "13688669282", "13688669283", "13688669284", "13688669285", "13688669286", "13688669287", "13688669288", "13688669289", "13688669290", "13688669291", "13688669292", "13688669293", "13688669294", "13688669295", "13688669296", "13688669297", "13688669298", "13688669299", "13688669300", "13688669301", "13688669302", "13688669303", "13688669304", "13688669305", "13688669306", "13688669307", "13688669308", "13688669309", "13688669310", "13688669311", "13688669312", "13688669313", "13688669314", "13688669315", "13688669316", "13688669317", "13688669318", "13688669319", "13688669320", "13688669321", "13688669322", "13688669323", "13688669324", "13688669325", "13688669326", "13688669327", "13688669328", "13688669329", "13688669330", "13688669331", "13688669332", "13688669333", "13688669334", "13688669335", "13688669336", "13688669337", "13688669338", "13688669339", "13688669340", "13688669341", "13688669342", "13688669343", "13688669344", "13688669345", "13688669346", "13688669347", "13688669348", "13688669349", "13688669350", "13688669351", "13688669352", "13688669353", "13688669354", "13688669355", "13688669356", "13688669357", "13688669358", "13688669359", "13688669360", "13688669361", "13688669362", "13688669363", "13688669364", "13688669365", "13688669366", "13688669367", "13688669368", "13688669369", "13688669370", "13688669371", "13688669372", "13688669373", "13688669374", "13688669375", "13688669376", "13688669377", "13688669378", "13688669379", "13688669380", "13688669381", "13688669382", "13688669383", "13688669384", "13688669385", "13688669386", "13688669387", "13688669388", "13688669389", "13688669390", "13688669391", "13688669392", "13688669393", "13688669394", "13688669395", "13688669396", "13688669397", "13688669398", "13688669399", "13688669400", "13688669401", "13688669402", "13688669403", "13688669404", "13688669405", "13688669406", "13688669407", "13688669408", "13688669409", "13688669410", "13688669411", "13688669412", "13688669413", "13688669414", "13688669415", "13688669416", "13688669417", "13688669418", "13688669419", "13688669420", "13688669421", "13688669422", "13688669423", "13688669424", "13688669425", "13688669426", "13688669427", "13688669428", "13688669429", "13688669430", "13688669431", "13688669432", "13688669433", "13688669434", "13688669435", "13688669436", "13688669437", "13688669438", "13688669439", "13688669440", "13688669441", "13688669442", "13688669443", "13688669444", "13688669445", "13688669446", "13688669447", "13688669448", "13688669449", "13688669450", "13688669451", "13688669452", "13688669453", "13688669454", "13688669455", "13688669456", "13688669457", "13688669458", "13688669459", "13688669460", "13688669461", "13688669462", "13688669463", "13688669464", "13688669465", "13688669466", "13688669467", "13688669468", "13688669469", "13688669470", "13688669471", "13688669472", "13688669473", "13688669474", "13688669475", "13688669476", "13688669477", "13688669478", "13688669479", "13688669480", "13688669481", "13688669482", "13688669483", "13688669484", "13688669485", "13688669486", "13688669487", "13688669488", "13688669489", "13688669490", "13688669491", "13688669492", "13688669493", "13688669494", "13688669495", "13688669496", "13688669497", "13688669498", "13688669499", "13688669500", "13688669501", "13688669502", "13688669503", "13688669504", "13688669505", "13688669506", "13688669507", "13688669508", "13688669509", "13688669510", "13688669511", "13688669512", "13688669513", "13688669514", "13688669515", "13688669516", "13688669517", "13688669518", "13688669519", "13688669520", "13688669521", "13688669522", "13688669523","13688669524", "13688669525", "13688669526", "13688669527", "13688669528", "13688669529", "13688669530", "13688669531", "13688669532", "13688669533", "13688669534", "13688669535", "13688669536", "13688669537", "13688669538", "13688669539", "13688669540", "13688669541", "13688669542", "13688669543", "13688669544", "13688669545", "13688669546", "13688669547", "13688669548", "13688669549", "13688669550", "13688669551", "13688669552", "13688669553", "13688669554", "13688669555", "13688669556", "13688669557", "13688669558", "13688669559", "13688669560", "13688669561", "13688669562", "13688669563", "13688669564", "13688669565", "13688669566", "13688669567", "13688669568", "13688669569", "13688669570", "13688669571", "13688669572", "13688669573", "13688669574", "13688669575", "13688669576", "13688669577", "13688669578", "13688669579", "13688669580", "13688669581", "13688669582", "13688669583", "13688669584", "13688669585", "13688669586", "13688669587", "13688669588", "13688669589", "13688669590", "13688669591", "13688669592", "13688669593", "13688669594", "13688669595", "13688669596", "13688669597", "13688669598", "13688669599", "13688669600", "13688669601", "13688669602", "13688669603", "13688669604", "13688669605", "13688669606", "13688669607", "13688669608", "13688669609", "13688669610", "13688669611", "13688669612", "13688669613", "13688669614", "13688669615", "13688669616", "13688669617", "13688669618", "13688669619", "13688669620", "13688669621", "13688669622", "13688669623", "13688669624", "13688669625", "13688669626", "13688669627", "13688669628", "13688669629", "13688669630", "13688669631", "13688669632", "13688669633", "13688669634", "13688669635", "13688669636", "13688669637", "13688669638", "13688669639", "13688669640", "13688669641", "13688669642", "13688669643", "13688669644", "13688669645", "13688669646", "13688669647", "13688669648", "13688669649", "13688669650", "13688669651", "13688669652", "13688669653", "13688669654", "13688669655", "13688669656", "13688669657", "13688669658", "13688669659", "13688669660", "13688669661", "13688669662", "13688669663", "13688669664", "13688669665", "13688669666", "13688669667", "13688669668", "13688669669", "13688669670", "13688669671", "13688669672", "13688669673", "13688669674", "13688669675", "13688669676", "13688669677", "13688669678", "13688669679", "13688669680", "13688669681", "13688669682", "13688669683", "13688669684", "13688669685", "13688669686", "13688669687", "13688669688", "13688669689", "13688669690", "13688669691", "13688669692", "13688669693", "13688669694", "13688669695", "13688669696", "13688669697", "13688669698", "13688669699", "13688669700", "13688669701", "13688669702", "13688669703", "13688669704", "13688669705", "13688669706", "13688669707", "13688669708", "13688669709", "13688669710", "13688669711", "13688669712", "13688669713", "13688669714", "13688669715", "13688669716", "13688669717", "13688669718", "13688669719", "13688669720", "13688669721", "13688669722", "13688669723", "13688669724", "13688669725", "13688669726", "13688669727", "13688669728", "13688669729", "13688669730", "13688669731", "13688669732", "13688669733", "13688669734", "13688669735", "13688669736", "13688669737", "13688669738", "13688669739", "13688669740", "13688669741", "13688669742", "13688669743", "13688669744", "13688669745", "13688669746", "13688669747", "13688669748", "13688669749", "13688669750", "13688669751", "13688669752", "13688669753", "13688669754", "13688669755", "13688669756", "13688669757", "13688669758", "13688669759", "13688669760", "13688669761", "13688669762", "13688669763", "13688669764", "13688669765", "13688669766", "13688669767", "13688669768", "13688669769", "13688669770", "13688669771", "13688669772", "13688669773", "13688669774", "13688669775", "13688669776", "13688669777", "13688669778", "13688669779", "13688669780", "13688669781", "13688669782", "13688669783", "13688669784", "13688669785", "13688669786", "13688669787", "13688669788", "13688669789", "13688669790", "13688669791", "13688669792", "13688669793", "13688669794", "13688669795", "13688669796", "13688669797", "13688669798", "13688669799", "13688669800", "13688669801", "13688669802", "13688669803", "13688669804", "13688669805", "13688669806", "13688669807", "13688669808", "13688669809", "13688669810", "13688669811", "13688669812", "13688669813", "13688669814", "13688669815", "13688669816", "13688669817", "13688669818", "13688669819", "13688669820", "13688669821", "13688669822", "13688669823", "13688669824", "13688669825", "13688669826", "13688669827", "13688669828", "13688669829", "13688669830", "13688669831", "13688669832", "13688669833", "13688669834", "13688669835", "13688669836", "13688669837", "13688669838", "13688669839", "13688669840", "13688669841", "13688669842", "13688669843", "13688669844", "13688669845", "13688669846", "13688669847", "13688669848","13688669849", "13688669849", "13688669850", "13688669851", "13688669852", "13688669853", "13688669854", "13688669855", "13688669856", "13688669857", "13688669858", "13688669859", "13688669860", "13688669861", "13688669862", "13688669863", "13688669864", "13688669865", "13688669866", "13688669867", "13688669868", "13688669869", "13688669870", "13688669871", "13688669872", "13688669873", "13688669874", "13688669875", "13688669876", "13688669877", "13688669878", "13688669879", "13688669880", "13688669881", "13688669882", "13688669883", "13688669884", "13688669885", "13688669886", "13688669887", "13688669888", "13838411438", "17359456898" }; for(String phone : phoneNumbers){ //一致根据手机号查用户 User user = query().eq("phone", phone).one(); //7.保存用户信息到redis---------------- //7.1 随机生成Token作为登录令牌 String token = UUID.randomUUID().toString(true); String filePath = "C:\\code\\output.txt"; String content = token+'\n'; try (FileWriter fileWriter = new FileWriter(filePath, true); BufferedWriter bufferedWriter = new BufferedWriter(fileWriter)) { // 写入内容 bufferedWriter.write(content); // 确保内容都已写入文件 bufferedWriter.flush(); } catch (IOException e) { throw new RuntimeException(e); } System.out.println(token); //7.2 将User对象转为Hash存储 UserDTO userDTO = BeanUtil.copyProperties(user, UserDTO.class); Map userMap = BeanUtil.beanToMap(userDTO,new HashMap<>(), CopyOptions.create() .setIgnoreNullValue(true) .setFieldValueEditor((fieldName,fieldValue)->fieldValue.toString())); //7.3 存储 stringRedisTemplate.opsForHash().putAll("login:token:"+token,userMap); //7.4设置token有效期 String tokenKey = LOGIN_USER_KEY+token; stringRedisTemplate.expire(tokenKey,999999999,TimeUnit.MINUTES); } }
更改秒杀库存为200:
订单表清空:
Jmeter中线程数设为1000:
在HTTP信息头管理器中进行如下设置:
在CSV数据文件设置中进行如下设置:
下面是测试结果:
最小值和最大值是响应时间的最小值和最大值。平均值是平均响应时间。
优惠券被抢完,没有超领和少领的情况发生:
刚好200条订单记录:
查询优惠券、查询订单、减库存、创建订单都需要与数据库交互,导致效率低下。特别是减库存和创建订单都是对数据库的写操作,耗时较久。
异步开启一个独立的线程去完成Tomcat的操作。
库存:KEY用string类型,VALUE用数值类型。
一人一单:KEY用string类型,VALUE用set集合类型。
因为这段代码比较长要用Lua脚本来编写:
首先要执行Lua脚本,然后判断返回结果是否为0,如返回0代表成功下单优惠券,将优惠券id、用户id和订单id放入阻塞队列,直接返回订单id给用户。
如果想提高写入数据库的性能,可以多开线程,由单个线程的写,变成多个线程批量的写。
在VoucherServiceImpl的addSeckillVoucher方法的末尾添加下面这段代码把秒杀的库存保存到Redis中:
//保存秒杀的库存到Redis stringRedisTemplate.opsForValue().set(RedisConstants.SECKILL_STOCK_KEY +voucher.getId(),voucher.getStock().toString());
发送请求,新增一份优惠券:
可以看到在Redis中记录了优惠券的记录:
在redis中可以用sadd来往set集合中添加键值,可以用sismember来查询集合中是否有某个元素。
Lua脚本编写如下:
--1.参数列表 --1.1.优惠券id local voucherId = ARGV[1] --1.2.用户id local userId = ARGV[2] --2.数据key --2.1.库存key local stockKey = 'seckill:stock:' .. voucherId --2.2.订单key local orderKey = 'seckill:order:' .. voucherId --3.脚本业务 --3.1.判断库存是否充足 get stockKey if(tonumber(redis.call('get',stockKey)) <= 0) then --3.1.2.库存不足,返回1 return 1 end --3.2.判断用户是否下单 SISMEMBER orderKey userId if(redis.call('sismember',orderKey,userId)==1) then --3.2.1.存在,说明是重复下单,返回2 return 2 end --3.3.扣库存 incrby stockKey -1 redis.call('incrby',stockKey,-1) --3.4.下单 sadd orderKey userId redis.call('sadd',orderKey,userId) return 0
在VoucherOrderServiceImpl类中写入如下代码:
Lua脚本的加载:
private static final DefaultRedisScript SECKILL_SCRIPT; static{ SECKILL_SCRIPT = new DefaultRedisScript<>(); SECKILL_SCRIPT.setLocation(new ClassPathResource("seckill.lua")); SECKILL_SCRIPT.setResultType(Long.class); }
执行Lua脚本:
Long result = stringRedisTemplate.execute( //调用execute方法,返回值 SECKILL_SCRIPT, //加载的模板对象 Collections.emptyList(), //键参数 voucherId.toString(), //值参数1 UserHolder.getUser().getId().toString() //值参数2 );
更改seckillVoucher方法的代码如下:
@Override public Result seckillVoucher(Long voucherId) { //1.执行Lua脚本 Long result = stringRedisTemplate.execute( SECKILL_SCRIPT, Collections.emptyList(), voucherId.toString(), UserHolder.getUser().getId().toString() ); int r = result.intValue(); if(r != 0){ //2.判断结果是否为0,不为0,代表没有购买资格 return Result.fail(r==1 ? "库存不足":"不能重复下单"); } //2.2.为0,有购买资格,把下单信息保存到阻塞队列 long orderId = redisIdWorker.nextId("order"); // TODO 保存阻塞队列 //3.返回订单id return Result.ok(orderId); }
在Apifox中发送测试数据,秒杀下单,成功后返回订单id:
在Redis中库存成功扣减1,order有缓存:
如果再次发送会提示不能重复下单:
准备在Jemeter中测试,首先把缓存中的优惠券库存改为200:
测试后库存减为0,新增200条订单记录:
可以看到平均响应时间减少10倍,最快响应时间减少60倍,最大响应时间缩短:
可见这种优化对系统的性能提升非常大!
阻塞队列:尝试从队列获取元素,如果没有元素会被阻塞,直到队列中有元素才会被唤醒,获取元素。
只要类一启动,用户随时都有可能来抢购,因此VoucherOrderHandler这个类的初始化必须在类初始化后执行。
在VoucherOrderServiceImpl类中,首先要新增一个orderTasks阻塞队列,然后设置一个线程池和run方法。
在run方法中调用阻塞队列的take方法,orderTasks.take方法是一个阻塞方法,如果队列中有元素会获取,如果队列中无元素则阻塞等待。
这里相当于是开启了一个全新的线程来执行获取队列中订单信息和异步创建订单的任务:
private BlockingQueue orderTasks = new ArrayBlockingQueue<>(1024*1024); private static ExecutorService seckill_order_executor = Executors.newSingleThreadExecutor(); @PostConstruct private void init(){ seckill_order_executor.submit(new VoucherOrderHandler()); } private class VoucherOrderHandler implements Runnable{ @SneakyThrows @Override public void run() { while(true){ try { //1.获取队列中的订单信息 VoucherOrder voucherOrder = orderTasks.take(); //2.创建订单 handleVoucherOrder(voucherOrder); } catch (InterruptedException e) { log.debug("处理订单异常",e); } } } }
然后新增一个handleVoucherOrder方法,这个方法主要用来获取锁然后调用createVoucherOrder方法:
public IVoucherOrderService proxy ; private void handleVoucherOrder(VoucherOrder voucherOrder) { //1.获取用户 Long userId = voucherOrder.getUserId(); //2.创建锁对象 RLock lock = redissonClient.getLock("lock:order:"+userId); //3.获取锁 boolean isLock = lock.tryLock(); //4.判断是否获取锁成功 if(!isLock) { log.error("不允许重复下单"); return; } try { //获取代理对象 proxy.createVoucherOrder(voucherOrder); }finally { lock.unlock(); } }
createVoucherOrder方法主要是用来对数据库操作,比如扣减库存,然后保存订单的信息到数据库,会有额外的对一人一单和库存数量的判断,虽然这些在Redis中已经判断过,但这里是双重保险。
异步处理不需要再返回给前端任何东西。
@Transactional public void createVoucherOrder(VoucherOrder voucherOrder) { //6.一人一单 Long userId = voucherOrder.getUserId(); //6.1查询订单 int count = query().eq("user_id", userId).eq("voucher_id", voucherOrder.getVoucherId()).count(); //6.2判断是否存在 if(count>0){ //用户已经购买过了 log.error("用户已经购买过一次!"); return; } //3.2库存充足扣减库存 boolean success = seckillVoucherService.update() .setSql("stock = stock - 1") //相当于set条件 set stock = stock - 1 .eq("voucher_id", voucherOrder.getVoucherId()) //相当于where条件 where id = ? and stock = ? .gt("stock",0).update(); if(!success){ log.error("库存不足!"); return; } long orderId = redisIdWorker.nextId("order");//订单id voucherOrder.setId(orderId); voucherOrder.setUserId(userId); voucherOrder.setVoucherId(voucherOrder.getVoucherId());//代金券id save(voucherOrder); }
下面是对seckillVoucher的简单修改:
public Result seckillVoucher(Long voucherId) { //1.执行Lua脚本 Long result = stringRedisTemplate.execute( SECKILL_SCRIPT, Collections.emptyList(), voucherId.toString(), UserHolder.getUser().getId().toString() ); int r = result.intValue(); if(r != 0){ //2.判断结果是否为0,不为0,代表没有购买资格 return Result.fail(r==1 ? "库存不足":"不能重复下单"); } //2.2.为0,有购买资格,把下单信息保存到阻塞队列 long orderId = redisIdWorker.nextId("order"); //封装 VoucherOrder voucherOrder = new VoucherOrder(); voucherOrder.setId(orderId);//订单id voucherOrder.setUserId(UserHolder.getUser().getId());//用户id voucherOrder.setVoucherId(voucherId);//代金券id //保存阻塞队列 orderTasks.add(voucherOrder); //获取代理对象 proxy = (IVoucherOrderService) AopContext.currentProxy(); //3.返回订单id return Result.ok(orderId); }
测试:
先把tb_voucher_order内容清空。把tb_seckill_voucher的stock库存改为200。
然后把Redis中对应优惠券的库存改为200。清空之前生成的订单。检查是否有1000个用户的token。
先用Apifox进行测试,测试一人一单的情况:第2次下单显示不能重复下单。
检查数据库是否多1条订单记录,库存是否减少1,缓存中库存是否减少1。
接下来用Jemeter进行测试,会发现库存扣减为0,数据库中多200条数据,缓存中的库存也扣减到0。
看聚合报告的结果如下:
因为做了异步下单,会占用一定的CPU,所以平均值要比第2次更长。
和下面前2次的结果进行对比可以发现,响应的平均值比最初提高10倍,最快响应时间提高了80倍,最慢响应时间提高了6倍。
秒杀业务的优化思路:
1.先利用Redis完成库存量、一人一单的判断,完成抢单业务。
2.将下单业务放入阻塞队列,利用独立线程异步下单。
基于阻塞队列的异步秒杀存在哪些问题:
1.内存限制问题。使用的是jdk提供的阻塞队列,使用的是JVM的内存,在一开始写死了队列空间的大小,如果在高并发的情况下,队列很快会被占满,如果不对队列的空间加以限制,很容易造成内存的溢出。
2.数据安全问题。缺乏持久化机制,是基于内存来保存信息,如果服务突然宕机,内存中保存的信息都会丢失。如果任务被取出,但由于突然发生事故异常,导致任务没有被消费,任务丢失,会造成数据不一致问题。
1.消息队列是在JVM外部的独立服务,不受JVM内存的限制。
2.消息队列不仅负责数据存储,还要保证数据安全。消息队列在消费者接收到消息后要进行消息确认
Redis提供了3种不同的方式来实现消息队列:
1.list结构:基于List结构模拟消息队列。
2.PubSub(发布订阅):基本的点对点消息模型。
3.Stream:比较完善的功能强大的消息队列模型。
Redis的list数据结构是一个双向链表,容易模拟出队列效果。
队列的入口和出口不在一边,可以利用:LPUSH结合RPOP,RPUSH结合LPOP来实现。
如果队列中没有消息时RPOP或LPOP的操作会返回null,不会像JVM的阻塞队列那样阻塞并等待消息,因此这里应该用BRPOP或BLPOP来实现阻塞效果。
List消息队列优点:
1.利用Redis存储,不受限于JVM内存上限。
2.基于Redis的持久化机制,数据安全性有保证。
3.可以保证消息的有序性。
缺点:
1.无法避免消息丢失。
2.只支持单消费者。
PubSub(Publish Subscribe 发布订阅):Redis2.0版本引入的消息传递模型,消费者可以订阅一个或多个channel,生产者向对应channel发送消息后,所有订阅者都能收到相关消息。
SUBSCRIBE 频道名称 : 订阅一个或多个频道。
PUBLISH channel msg : 向一个频道发送消息。
PSUBSCRIBE pattern : 订阅与pattern格式匹配的所有频道。
基于PubSub的消息队列有哪些优缺点:
优点:
1.采用发布订阅模型,支持多生产、多消费。
缺点:
1.不支持数据持久化。
2.无法避免消息丢失(如果发布的消息没有人订阅,消息直接丢失)安全性无法保障
3.消息堆积有上限(消费者缓存的空间有上限),超出时数据丢失。
要注意,如果想使用Stream消息队列必须把Redis的版本上升到5.0之后。
需要注意的是key和*|ID中间那俩参数是可选参数,一个是用来判断是否自动创建队列,一个是用来设置队列最大消息数量。
当指定起始ID为$时,代表读取最新的消息,如果我们处理一条消息的过程中,又有超过1条以上的消息到达队列,则下次读取的还是最新的一条消息,此时中间的几条消息会被漏读。
STREAM类型消息队列的XREAD命令特点:
1.消息可回溯。
2.一个消息可以被多个消费者读取。
3.可以阻塞读取。
4.有消息漏读风险。
消费者组(Consumer Group):将多个消费者划分到一个组中,监听同一个队列。消费者之间是竞争关系。
1.消息分流:队列中的消息会分流给组内不同消费者,而不是重复消费,从而加快消息处理的速度。
2.消息标示:消费者组会维护一个标示(类似于标签,记录读到哪里了),记录最后一个被处理的消息,哪怕消费者宕机重启,还会从标示之后读取消息,确保每一个消息都会被消费。
3.消息确认(解决消息丢失问题):消费者获取消息后,消息处于pending状态,并存入一个pending-list。当处理完成后需要通过XACK来确认消息,标记消息为已处理,才会从pending-list移除。
创建消费者组:
XGROUP CREATE key groupName ID [MKSTREAM]
key是队列名称
groupName是消费者组名称
ID是起始ID标示,$代表队列最后一个消息,0代表队列第一个消息。
MKSTREAM是队列不存在时自动创建队列。
删除指定的消费者组:
XGROUP DESTORY key groupName
给指定的消费者组添加消费者:
XGROUP CREATECONSUMER key groupname consumername
删除消费者组中的指定消费者:
XGROUP DELCONSUMER key groupname consumername
创建消费者组:
从消费者组读取消息:
group:是消费者组名称。
consumer:是消费者名称,如果消费者不存在,会自动创建。
count:本次查询最大数量。
BLOCK:阻塞时最长等待时间。
NOACK:取消消费者手动ACK,获取到消息自动确认(不建议开启)
STREAMS key:指定队列名称
ID:获取消息的起始ID。(">":从下一个未消费的消息开始读取。其它:根据指定id从pending-list中获取已消费但未确认的消息,例如0,是从pending-list中的第一个消息开始)
可以发现在同一个消费者组里的消费者对消息不会重复读取,而是依次读取,已被读取的消息不会再次被读取。
消费者确认消息:
XACK key group ID
key:是队列名称。
group:是消费者组名称。
ID:是接收到的消息的ID。
查看Pending-list队列的信息:
key:是队列名称。
group:是组名称。
下面是消息确认:
对消息进行确认,确认完消息会被移除:
Pending-list队列里面存储的是已经读取,但是还没确认的消息。
假如一台节点读取完消息还没却来得及确认就宕机了,可以通过以下的方法解决:
正常情况下先用>,如果出现异常,信息会进入到Pending-list,把ID从>改为0,此时取的就是在Pending-list里的消息。
STREAM类型消息队列的XREADGROUP命令特点:
1.消息可回溯。
2.可以多消费者争抢消息,加快消费速度。
3.可以阻塞读取。
4.没有消息漏读的风险。
5.有消息确认机制,消息至少被消费一次。
6.支持消息持久化
如果公司业务比较庞大,对于消息队列的要求更加严格,还是要用RabbitMQ和RocketMQ。
直接通过控制台创建一个stream.orders队列:
直接在Lua脚本中编写代码(主要增加一个局部变量,):
--1.参数列表 --1.1.优惠券id local voucherId = ARGV[1] --1.2.用户id local userId = ARGV[2] --1.3.订单id local orderId = ARGV[3] --2.数据key --2.1.库存key local stockKey = 'seckill:stock:' .. voucherId --2.2.订单key local orderKey = 'seckill:order:' .. voucherId --3.脚本业务 --3.1.判断库存是否充足 get stockKey if(tonumber(redis.call('get',stockKey)) <= 0) then --3.1.2.库存不足,返回1 return 1 end --3.2.判断用户是否下单 SISMEMBER orderKey userId if(redis.call('sismember',orderKey,userId)==1) then --3.2.1.存在,说明是重复下单,返回2 return 2 end --3.3.扣库存 incrby stockKey -1 redis.call('incrby',stockKey,-1) --3.4.下单 sadd orderKey userId redis.call('sadd',orderKey,userId) --3.5.发送消息到队列中 XADD stream.orders * k1 v1 k2 v2 redis.call('xadd','stream.orders','*','userId',userId,'voucherId',voucherId,'orderId',orderId) return 0
在VoucherOrderServiceImpl类中修改seckillVoucher方法:
public Result seckillVoucher(Long voucherId) { //获取订单id long orderId = redisIdWorker.nextId("order"); //1.执行Lua脚本(判断用户是否有购买资格,消息发出) Long result = stringRedisTemplate.execute( SECKILL_SCRIPT, Collections.emptyList(), voucherId.toString(), UserHolder.getUser().getId().toString(), String.valueOf(orderId) ); int r = result.intValue(); if(r != 0){ //2.判断结果是否为0,不为0,代表没有购买资格 return Result.fail(r==1 ? "库存不足":"不能重复下单"); } //获取代理对象 proxy = (IVoucherOrderService) AopContext.currentProxy(); //3.返回订单id return Result.ok(orderId); }
在VoucherOrderServiceImpl中修改VoucherOrderHandler方法的代码:
代码思路如下:
1.从消息队列中尝试读消息。
1.1.获取失败,继续循环。
2.获取成功,进行解析和转换。
3.调用createVoucherOrder(voucherOrder)方法完成下单。
4.ACK确认
4.1.确认失败,调用handlePendingList()方法进行处理。
private static ExecutorService seckill_order_executor = Executors.newSingleThreadExecutor(); @PostConstruct private void init(){ seckill_order_executor.submit(new VoucherOrderHandler()); } private class VoucherOrderHandler implements Runnable{ String queueName = "stream.order"; @SneakyThrows @Override public void run() { while(true){ try { //1.获取消息队列中的订单信息 XREADGROUP GROUP g1 c1 COUNT 1 BLOCK 2000 STREAMS streams.order > List> list = stringRedisTemplate.opsForStream().read( Consumer.from("g1", "c1"), StreamReadOptions.empty().count(1).block(Duration.ofSeconds(2)), StreamOffset.create(queueName, ReadOffset.lastConsumed()) ); //2.判断消息获取是否成功 if(list==null || list.isEmpty()){ //2.1.获取失败,没有消息,继续下一次循环 continue; } //3.解析消息中的订单信息 MapRecord record = list.get(0); //4.获取成功,可以下单 Map
在VoucherOrderServiceImpl中添加handlePendingList()方法的代码:
下面有几个修改点:1.XREADGROUP语句末尾改为0,表示读Pending-list队列。2.Pending-list消息获取失败结束循环。3.如果抛异常只是暂停一下,然后会继续循环读。
private void handlePendingList() { while(true){ try { //1.获取Pending-List中的订单信息 XREADGROUP GROUP g1 c1 COUNT 1 BLOCK 2000 STREAMS streams.order 0 List> list = stringRedisTemplate.opsForStream().read( Consumer.from("g1", "c1"), StreamReadOptions.empty().count(1), StreamOffset.create(queueName, ReadOffset.from("0")) ); //2.判断消息获取是否成功 if(list==null || list.isEmpty()){ //2.1.获取失败,说明Pending-list里没有异常消息,结束循环 break; } //3.解析消息中的订单信息 MapRecord record = list.get(0); //4.获取成功,可以下单 Map
启动项目进行测试:
首先用Apifox进行测试,测试接口请求发送成功:
测试成功后可以看到:tb_voucher_order表多了1条记录,tb_seckill_voucher表对应优惠券的库存-1;在Redis中seckill:order下出现订单记录,在stockill:stock下的库存-1,在stream.orders下出现1条新的记录。
然后用Jmeter进行测试:可以发现相较于未做异步处理的情况性能仍有较大提升。
一般来讲企业开发会将图片等文件上传到一个专门的文件服务器上。
但我们这个项目目前只会将文件上传到前端服务器上。
复制下面的链接,然后保存到下面这个位置:
上传图片:
在我的和首页都可以看到新发布的博文:
探店笔记要包含笔记的内容和博主的相关信息。所以选择在Blog表中添加如下2个字段,这两个字段需要后续我们手动维护(赋值)。
下面是接口的请求地址和说明,建议先不看视频自己写,写完后和视频比较差异。
在BlogController类里添加一个queryBlogById方法(虽然Controller里不应该出现业务代码,但鉴于只是简单的查询操作,就不必在意细节了):
@GetMapping("/{id}") public Result queryBlogById(@PathVariable("id") Long id){ Blog blog = blogService.getById(id); if(blog==null){ return Result.fail("笔记不存在"); } User user = userService.getById(id); blog.setIcon(user.getIcon()); blog.setName(user.getNickName()); return Result.ok(blog); }
现在的点赞逻辑是,一个人可以对同一篇笔记点赞无数次。
需求:
1.同一个用户只能点赞一次,如果再次点击则取消点赞。
2.如果当前用户已经点赞,则点赞按钮高亮显示(前端已实现,判断字段Blog类的isLike属性)
分析:
1.给Blog类中添加一个isLike字段,标示是否被当前用户点赞。
2.修改点赞功能,利用Redis的Set集合判断是否点赞过,未点赞则点赞数+1,已点赞则点赞数-1。
3.修改分页查询Blog业务和根据id查询Blog的业务,判断当前用户是否点赞过,赋值给isLike字段。
代码如下:
在BlogController类中新增likeBlog方法:
@PutMapping("/like/{id}") public Result likeBlog(@PathVariable("id") Long id) { return blogService.likeBlog(id); }
在IBlogService接口中添加方法声明:
Result likeBlog(Long id);
在BlogServiceImpl类中添加下面代码:
private final StringRedisTemplate stringRedisTemplate; public BlogServiceImpl(StringRedisTemplate stringRedisTemplate) { this.stringRedisTemplate = stringRedisTemplate; } @Override public Result likeBlog(Long id) { //1.获取登录用户 UserDTO user = UserHolder.getUser(); Long userId = user.getId(); //2.判断当前用户是否已经点赞过 String key = "blog:liked:" +id; Boolean isMember = stringRedisTemplate.opsForSet().isMember(key, userId.toString()); if(BooleanUtil.isFalse(isMember)){ //3.未点赞,可以点赞 //3.1.数据库点赞数+1 boolean isSuccess = update().setSql("liked=liked+1").eq("id", id).update(); //3.2.保存用户到Redis if(isSuccess){ stringRedisTemplate.opsForSet().add(key,userId.toString()); } }else{ //4.已点赞,取消点赞 //4.1.数据库点赞数-1 boolean isSuccess = update().setSql("liked=liked-1").eq("id", id).update(); //4.2.把用户从Redis的set集合移除 stringRedisTemplate.opsForSet().remove(key,userId.toString()); } return Result.ok(); }
因为我的queryBlogById和queryHotBlog的业务代码都沿用原本的代码写在BlogController中,因此我是直接在BlogController中写入isBlogLiked代码:
public Boolean isBlogLiked(Blog blog) { Long userId = null; try { //1.获取登录用户 userId = UserHolder.getUser().getId(); } catch (Exception e) { log.debug("用户未登录!"); return false; } //2.判断当前用户是否已经点赞过 String key = "blog:liked:" +blog.getId(); Boolean isMember = stringRedisTemplate.opsForSet().isMember(key, userId.toString()); try { blog.setIsLike(BooleanUtil.isTrue(isMember)); } catch (Exception e) { log.debug("点赞信息为空!"); return false; } return isMember; }
isBlogLiked的作用主要是给blog对象设置值(在Java中对象是引用传递),前端会根据返回的blog对象的isLike参数的true或false来给点赞标签高亮或灰暗。
测试效果:点赞一次高亮,点赞两次取消。在缓存中有相应的记录:
需求:在探店笔记详情页面,按照时间排序,把最早点赞的TOP5列举出来,形成点赞排行榜。
我们选用SortedSet来实现功能。
可以用ZADD命令添加元素,ZSCOPE来获得分数对应的元素,ZRANGE来
修改的点有如下几个:
在Redis缓存中多了一个score:
在BlogController类中添加如下方法:
@GetMapping("/likes/{id}") public Result queryBlogLikes(@PathVariable("id") Long id) { return blogService.queryBlogLikes(id); }
在IBlogService接口中添加如下方法:
Result queryBlogLikes(Long id);
在BlogServiceImpl类中添加如下方法:
@Override public Result queryBlogLikes(Long id) { String key = RedisConstants.BLOG_LIKED_KEY +id; //1.查询top5的点赞用户 zrange key 0 4 Set top5 = stringRedisTemplate.opsForZSet().range(key, 0, 4); if(top5==null || top5.isEmpty()){ return Result.ok(Collections.emptyList()); } //2.解析出其中的用户id List ids = top5.stream().map(Long::valueOf).collect(Collectors.toList()); //3.根据用户id查询用户 List userDTOS = userService.listByIds(ids) .stream() .map(user -> BeanUtil.copyProperties(user, UserDTO.class)) .collect(Collectors.toList()); //4.返回 return Result.ok(userDTOS); }
现在会出现左图问题,先点赞的反而被排到后面了:
下面是修改后的queryBlogLikes:
@Override public Result queryBlogLikes(Long id) { String key = RedisConstants.BLOG_LIKED_KEY +id; //1.查询top5的点赞用户 zrange key 0 4 Set top5 = stringRedisTemplate.opsForZSet().range(key, 0, 4); if(top5==null || top5.isEmpty()){ return Result.ok(Collections.emptyList()); } //2.解析出其中的用户id List ids = top5.stream().map(Long::valueOf).collect(Collectors.toList()); //3.根据用户id查询用户 String idStr = StrUtil.join(",", ids); List userDTOS = userService.query() .in("id",ids) .last("ORDER BY FIELD(id,"+idStr+")").list() .stream() .map(user -> BeanUtil.copyProperties(user, UserDTO.class)) .collect(Collectors.toList()); //4.返回 return Result.ok(userDTOS); }
能够正常展示:
关注和取关请求:
查看是否已关注:
关注是给表新增记录,取关是删除表中记录。
在FollowController类中写入如下代码:
@RestController @RequestMapping("/follow") public class FollowController { @Autowired IFollowService followService; @PutMapping("/{id}/{isFollow}") public Result follow(@PathVariable("id") Long followUserId, @PathVariable("isFollow") Boolean isFollow) { return followService.follow(followUserId,isFollow); } @GetMapping("/or/not/{id}") public Result isFollow(@PathVariable("id") Long followUserId) { return followService.isFollow(followUserId); } }
在IFollowService中写入如下代码:
public interface IFollowService extends IService { Result follow(Long followUserId, Boolean isFollow); Result isFollow(Long followUserId); }
在FollowServiceImpl类中写入如下代码:
@Service public class FollowServiceImpl extends ServiceImpl implements IFollowService { @Override public Result follow(Long followUserId, Boolean isFollow) { Long userId = UserHolder.getUser().getId(); //1.判断是关注还是取关 if(isFollow){ //2.关注,新增数据 Follow follow = new Follow(); follow.setUserId(userId); follow.setFollowUserId(followUserId); save(follow); }else{ //3.取关,删除记录 remove(new QueryWrapper() .eq("user_id", userId).eq("follow_user_id", followUserId)); } return Result.ok(); } @Override public Result isFollow(Long followUserId) { Long userId = UserHolder.getUser().getId(); //1.查询是否关注 Integer count = query().eq("user_id", userId).eq("follow_user_id", followUserId).count(); //2.判断是否关注 return Result.ok(count>0); } }
测试:点击关注显示关注成功,数据库里会有记录。
取消关注后记录消息。
把准备好的queryUserById方法放入到UserController中
把准备好的queryBlogByUserId方法放入到BlogController中
功能实现的思路是:在关注某位用户之后,同时将被关注的用户id存入到Redis中。查看共同关注的时候,只需要求被查看的这位用户与自己关注列表的交集即可。
首先要更改FollowServiceImpl代码中的follow方法,主要在关注时把被关注用户的id放入redis,取关时从redis中移除id:
@Resource private final StringRedisTemplate stringRedisTemplate; @Resource private IUserService userService; public FollowServiceImpl(StringRedisTemplate stringRedisTemplate) { this.stringRedisTemplate = stringRedisTemplate; } @Override public Result follow(Long followUserId, Boolean isFollow) { Long userId = UserHolder.getUser().getId(); String key = "follows:" + userId; //1.判断是关注还是取关 if(isFollow){ //2.关注,新增数据 Follow follow = new Follow(); follow.setUserId(userId); follow.setFollowUserId(followUserId); boolean isSuccess = save(follow); if(isSuccess){ //把关注用户的id放入redis的set集合 stringRedisTemplate.opsForSet().add(key,followUserId.toString()); } }else{ //3.取关,删除记录 boolean isSuccess = remove(new QueryWrapper() .eq("user_id", userId).eq("follow_user_id", followUserId)); if(isSuccess){ //把关注用户的id从Redis移除 stringRedisTemplate.opsForSet().remove(key,followUserId.toString()); } } return Result.ok(); }
下面进行简单测试,关注时数据存入redis没问题:
点击共同关注报错,但发出了请求,原因是还没编写方法:
可以通过SINTER命令求出交集:
在FollowController中写入下面代码:
@GetMapping("/common/{id}") public Result followCommons(@PathVariable("id") Long id){ return followService.followCommons(id); }
在IFollowService中写入下面代码:
Result followCommons(Long id);
在FollowServiceImpl类中写入下面代码:
@Override public Result followCommons(Long id) { //求的是目标用户和当前用户关注的交集 //1.获取key Long userId = UserHolder.getUser().getId(); String key1 = "follows:"+userId; String key2 = "follows:"+id; //2.求交集 Set intersect = stringRedisTemplate.opsForSet().intersect(key1, key2); if(intersect==null||intersect.isEmpty()){ return Result.ok(Collections.emptyList()); } //3.解析id集合 List ids = intersect.stream().map(Long::valueOf).collect(Collectors.toList()); //4.查询用户 List users = userService.listByIds(ids) .stream() .map(user -> BeanUtil.copyProperties(user, UserDTO.class)) .collect(Collectors.toList()); return Result.ok(users); }
像我现在关注的是小鱼同学和可可今天不吃肉。接下来换号,换成小鱼同学,电话:13686869696,让小鱼同学关注我和可可今天不吃肉。以小鱼同学的视角来查看我,可以看到我们共同关注了可可今天不吃肉。
关注推送也叫做Feed流,直译为投喂。为用户持续的提供“沉浸式”的体验,通过无限下拉刷新获取新的消息。
Feed流产品有2种常见模式:
1.Timeline:不做内容筛选,简单的按照内容发布时间排序,常用于好友或关注。例如朋友圈。
优点:信息全面,不会有缺失。并且实现也相对简单。
缺点:信息噪音较多,用户不一定感兴趣,内容获取效率低。
2.智能排序:利用智能算法屏蔽掉违规的、用户不感兴趣的内容。推送用户感兴趣的信息来吸引用户。
优点:投喂用户感兴趣的信息,用户粘度高,容易沉迷。
缺点:如果算法不精确,可能起反作用。
— — — — — — — — — — — — — — — —
1.拉模式:读扩散。
信息的发送端会把信息发送到发件箱。等信息的接受端要读信息时,把发件箱的消息拉取到收件箱,然后将所有消息按照时间进行排序。
缺点:操作耗费时间多,存在延迟。
2.推模式:写扩散。
信息的发送端会直接把信息发送到所有接收方的收件箱。接收方收件箱内的消息按时间逐个进行排序。
缺点:需要保存大量的消息。
3.推拉结合模式:读写混合,兼具推和拉两种模式的优点。
当普通人发消息,因为粉丝少,直接采用推模式。
当大V发消息,因为粉丝多,对于活跃粉丝(数量少)采用推模式,对于普通粉丝僵死粉(数量多)采用拉模式。
要实现分页查询要指定page和size,计算从哪里开始到哪里结束。
List有角标,SortedSet有排名。
Feed流中的数据会不断更新,所以数据的角标也在变化,因此不能用传统的分页模式。
用滚动分页模式:
score范围进行查询,记住最小的时间戳,下次找比这个更小的时间戳。
在BlogController里面写入如下代码:
@PostMapping public Result saveBlog(@RequestBody Blog blog) { // 获取登录用户 UserDTO user = UserHolder.getUser(); blog.setUserId(user.getId()); // 保存探店博文 blogService.saveBlog(blog); // 返回id return Result.ok(blog.getId()); }
在IBlogService接口里面写入如下代码:
Result saveBlog(Blog blog);
在BlogServiceImpl类中写入如下代码:
@Override public Result saveBlog(Blog blog) { //1.获取登录用户 UserDTO user = UserHolder.getUser(); blog.setUserId(user.getId()); //2.保存探店笔记 boolean isSuccess = save(blog); if(!isSuccess){ return Result.fail("新增笔记失败!"); } //3.查询笔记作业的所有粉丝 //select * from tb_follow where follow_user_id = ? List follows = followService.query().eq("follow_user_id", user.getId()).list(); //4.推送笔记id给粉丝 for(Follow follow : follows){ //4.1.获取粉丝id Long userId = follow.getUserId(); //4.2.推送到粉丝收件箱是sortedSet String key = "feed::"+userId; stringRedisTemplate.opsForZSet().add(key,blog.getId().toString(),System.currentTimeMillis()); } //返回id return Result.ok(blog.getId()); }
ZRANGE是按照角标从小到大排序:
ZREVRANGE是按照角标从大到小排序:
ZREVRANGEBYSCORE是按照分数从大到小排序:
滚动查询:每一次都记住上一次查询分数的最小值,将最小值作为下一次的最大值
分数的最大值,分数的最小值,偏移量,查的数量。
规律:分数最小值和查的数量固定不变。最大值为上一次查询的最小值、偏移量第1次给0,第1次后给在上一次的结果中,与最小值一样的元素的个数。
当分数一致出现问题:
在BlogController中写入下面代码:
@GetMapping("/of/follow") public Result queryBlogOfFollow(@RequestParam("lastId") Long max,@RequestParam(value = "offset",defaultValue = "0") Integer offset) { return blogService.queryBlogOfFollow(max,offset); }
在IBlogService写入如下代码:
Result queryBlogOfFollow(Long max, Integer offset);
在BlogServiceImpl中写入如下代码:
@Override public Result queryBlogOfFollow(Long max, Integer offset) { //1.获取当前用户 Long userId = UserHolder.getUser().getId(); //2.查询收件箱 ZREVRANGEBYSCORE key Max Min LIMIT offset count String key = FEED_KEY+userId; Set> typedTuples = stringRedisTemplate.opsForZSet() .reverseRangeByScoreWithScores(key, 0, max, offset, 3); //2.1.非空判断 if(typedTuples==null || typedTuples.isEmpty()){ return Result.ok(); } //3.解析数据:blogId、minTime(时间戳)、offset List ids = new ArrayList<>(typedTuples.size()); long minTime = 0; int os = 1; for(ZSetOperations.TypedTuple tuple:typedTuples){ //4.1.获取id String idStr = tuple.getValue(); ids.add(Long.valueOf(idStr)); //4.2.获取分数 long time = tuple.getScore().longValue(); if(time == minTime){ os++; }else{ minTime = time; os=1; } } //4.根据id查询blog String idStr = StrUtil.join(",",ids); List blogs = query().in("id",ids).last("ORDER BY FIELD(id,"+idStr+")").list(); for (Blog blog : blogs) { isBlogLiked(blog); User user = userService.getById(blog.getUserId()); blog.setName(user.getNickName()); blog.setIcon(user.getIcon()); } //5.封装并返回 ScrollResult r = new ScrollResult(); r.setList(blogs); r.setOffset(os); r.setMinTime(minTime); return Result.ok(r); }
效果图如下(我关注了小鱼同学,于是我可以看到小鱼同学发布的文章):
GEO是Geolocation的简写形式,代表地理坐标,在Redis的3.2版本后加入了GEO的支持,允许存储地理坐标信息,帮助我们根据经纬度来检索数据。
在src/test/java/com/hmdp的HmDianPingApplicationTests类中写入如下的方法:
@Resource StringRedisTemplate stringRedisTemplate; @Test void loadShopData(){ //1.查询店铺信息 List list = shopService.list(); //2.把店铺分组,按照typeId分组,typeId一致的放到一个集合 Map> map = list.stream().collect(Collectors.groupingBy(Shop::getTypeId)); //3.分批完成写入Redis for (Map.Entry> entry : map.entrySet()) { //3.1.获取类型id Long typeid = entry.getKey(); String key = "shop:geo:"+typeid; //3.2.获取同类型的店铺的集合 List value = entry.getValue(); //3.3.写入redis GEOADD key 经度 纬度 member List> locations = new ArrayList<>(value.size()); for(Shop shop : value){ // stringRedisTemplate.opsForGeo().add(key,new Point(shop.getX(),shop.getY()),shop.getId().toString()); locations.add(new RedisGeoCommands.GeoLocation<>( shop.getId().toString(), new Point(shop.getX(),shop.getY()) )); } stringRedisTemplate.opsForGeo().add(key,locations); } }
点击运行之后,在Redis中能够看到导入的数据:
首先排除掉spring-data-redis和lettuce-core这俩依赖,然后引入这俩依赖的新版本:
org.springframework.boot spring-boot-starter-data-redis spring-data-redis org.springframework.data lettuce-core io.lettuce org.springframework.data spring-data-redis 2.6.2 compile io.lettuce lettuce-core 6.1.6.RELEASE compile
在ShopController中修改queryShopByType方法:
@GetMapping("/of/type") public Result queryShopByType( @RequestParam("typeId") Integer typeId, @RequestParam(value = "current", defaultValue = "1") Integer current, @RequestParam(value="x",required=false) Double x, @RequestParam(value="y",required=false) Double y ) { return shopService.queryShopByType(typeId,current,x,y); }
在IShopService接口中写入如下方法:
Result queryShopByType(Integer typeId, Integer current, Double x, Double y);
在ShopServiceImpl类中写入如下方法:
@Override public Result queryShopByType(Integer typeId, Integer current, Double x, Double y) { //1.判断是否需要根据坐标查询 if(x==null || y==null){ //不需要查询坐标,按数据库查 Page page = query() .eq("type_id",typeId) .page(new Page<>(current, SystemConstants.DEFAULT_PAGE_SIZE)); return Result.ok(page.getRecords()); } //2.计算分页参数 int from = (current - 1)*SystemConstants.DEFAULT_PAGE_SIZE; int end = current*SystemConstants.DEFAULT_PAGE_SIZE; //3.查询redis,按照距离排序、分页。结果:shopId,distance String key = SHOP_GEO_KEY+typeId; GeoResults> results = stringRedisTemplate.opsForGeo() .search( key, GeoReference.fromCoordinate(x, y), new Distance(5000), RedisGeoCommands.GeoSearchCommandArgs.newGeoSearchArgs().includeDistance().limit(end) ); //4.解析出id if(results==null){ return Result.ok(Collections.emptyList()); } List>> list = results.getContent(); //4.1.截取from-end的部分 List ids = new ArrayList<>(list.size()); Map distanceMap = new HashMap<>(list.size()); if(list.size()<=from){ return Result.ok(Collections.emptyList()); } list.stream().skip(from).forEach(result->{ //跳过可能把所有数据跳过了 //4.2.获取店铺id String shopIdStr = result.getContent().getName(); ids.add(Long.valueOf(shopIdStr)); //4.3.获取距离 Distance distance = result.getDistance(); distanceMap.put(shopIdStr,distance); }); //5.根据id查询shop String idStr = StrUtil.join(",", ids); List shops = query().in("id", ids).last("ORDER BY FIELD ( id," + idStr + ")").list(); for(Shop shop : shops){ shop.setDistance(distanceMap.get(shop.getId().toString()).getValue()); } //6、返回 return Result.ok(shops); }
效果:
BITFIELD能一次查询多个比特位的值。
BITFIELD key GET(代表查询) u(u代表无符号,i代表有符号)截取几位作为结果 开始的位置
在UserController中写入下面的方法:
@PostMapping("/sign") public Result sign(){ return userService.sign(); }
在IUserService中写入下面的方法:
Result sign();
在UserServiceImpl类中写入如下代码:
@Override public Result sign() { //1.获取当前登录用户 Long userId = UserHolder.getUser().getId(); //2.获取日期 LocalDateTime now = LocalDateTime.now(); //3.拼接key String keySuffix = now.format(DateTimeFormatter.ofPattern("yyyyMM")); String key = USER_SIGN_KEY + userId + keySuffix; //4.获取今天是本月第几天 int dayOfMonth = now.getDayOfMonth(); //5.写入Redis SETBIT key offset 1 stringRedisTemplate.opsForValue().setBit(key,dayOfMonth-1,true); return Result.ok(); }
用Apifox进行测试:
在UserController类中写入如下代码:
@GetMapping("/sign/count") public Result signCount(){ return userService.signCount(); }
在IUserService类中写入如下代码:
Result signCount();
在UserServiceImpl类中写入如下代码:
@Override public Result signCount() { //1.获取当前登录用户 Long userId = UserHolder.getUser().getId(); //2.获取日期 LocalDateTime now = LocalDateTime.now(); //3.拼接key String keySuffix = now.format(DateTimeFormatter.ofPattern("yyyyMM")); String key = USER_SIGN_KEY + userId + keySuffix; //4.获取今天是本月第几天 int dayOfMonth = now.getDayOfMonth(); //5.获取本月截止今天为止的所有的签到记录,返回的是一个十进制的数字 BITFIELD sign:5:202203 GET u14 0 List result = stringRedisTemplate.opsForValue().bitField( key, BitFieldSubCommands.create() .get(BitFieldSubCommands.BitFieldType.unsigned(dayOfMonth)).valueAt(0) ); if(result==null || result.isEmpty()){ //没有任何签到结果 return Result.ok(0); } Long num = result.get(0); if(num==null || num==0){ return Result.ok(0); } //6.循环遍历 int count=0; while(true){ //6.1.让这个数字与1做与运算,得到数字的最后一个bit位 if((num&1)==0){//6.2.判断这个bit位是否为0 //6.3.如果为0,说明未签到结束 break; }else{ //6.4.如果不为0,说明已签到,计数器+1 count++; } //6.5.把数字右移一位,抛弃最后一个bit位,继续下一个bit位 num >>>= 1; } return Result.ok(count); }
不论添加几次,永远只记录一次。
在src/test/java/com/hmdp的HmDianPingApplicationTests类中写入如下代码:
@Test void testHyperLogLog(){ String[] values = new String[1000]; int j=0; for(int i=0;i<1000000;i++){ j=j%1000; values[j] = "user_"+i; if(j == 999){ stringRedisTemplate.opsForHyperLogLog().add("hl3",values); } j++; } //统计数量 Long res = stringRedisTemplate.opsForHyperLogLog().size("hl3"); System.out.println("hl3"+res); }
100万数据成功写入:
测试前:
测试后:
大约占用11kb,确实是小于16kb。
备忘录:
tasklist | findstr nginx
taskkill /F /PID 47096
netstat -aon | findstr :8080
在我电脑上Redis是在C:\cangqiongwaimai\Redis-x64-3.2.100\redis-server.exe里。
Jmeter是在C:\software\apache-jmeter-5.5\bin\jmeter.bat里。