当前位置: 首页 > news >正文

导航网站 wordpress网络营销师资格证报名

导航网站 wordpress,网络营销师资格证报名,企业网站备案要求,有趣的网站初音Redis 学习笔记 3#xff1a;黑马点评 准备工作 需要先导入项目相关资源#xff1a; 数据库文件 hmdp.sql后端代码 hm-dianping.zip包括前端代码的 Nginx 启动后端代码和 Nginx。 短信登录 发送验证码 PostMapping(code) public Result sendCode(RequestP…Redis 学习笔记 3黑马点评 准备工作 需要先导入项目相关资源 数据库文件 hmdp.sql后端代码 hm-dianping.zip包括前端代码的 Nginx 启动后端代码和 Nginx。 短信登录 发送验证码 PostMapping(code) public Result sendCode(RequestParam(phone) String phone, HttpSession session) {// 发送短信验证码并保存验证码return userService.sendCode(phone, session); }Log4j2 Service public class UserServiceImpl extends ServiceImplUserMapper, User implements IUserService {Overridepublic Result sendCode(String phone, HttpSession session) {if (RegexUtils.isPhoneInvalid(phone)) {return Result.fail(不是合法的手机号);}String code RandomUtil.randomNumbers(6);session.setAttribute(code, code);// 发送短信log.debug(发送短信验证码{}, code);return Result.ok();} }登录 PostMapping(/login) public Result login(RequestBody LoginFormDTO loginForm, HttpSession session) {// 实现登录功能return userService.login(loginForm, session); }Override public Result login(LoginFormDTO loginForm, HttpSession session) {// 验证手机号和验证码if (RegexUtils.isPhoneInvalid(loginForm.getPhone())) {return Result.fail(手机号不合法);}String code (String) session.getAttribute(code);if (code null || !code.equals(loginForm.getCode())) {return Result.fail(验证码不正确);}// 检查用户是否存在QueryWrapperUser qw new QueryWrapper();qw.eq(phone, loginForm.getPhone());User user this.baseMapper.selectOne(qw);if (user null) {user this.createUserByPhone(loginForm.getPhone());}// 将用户信息保存到 sessionsession.setAttribute(user, user);return Result.ok(); }private User createUserByPhone(String phone) {User user new User();user.setPhone(phone);user.setNickName(user_ RandomUtil.randomString(5));this.baseMapper.insert(user);return user; }统一身份校验 定义拦截器 public class LoginInterceptor implements HandlerInterceptor {Overridepublic 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;}// 将用户信息保存到 ThreadLocalUserDTO userDTO new UserDTO();userDTO.setIcon(user.getIcon());userDTO.setId(user.getId());userDTO.setNickName(user.getNickName());UserHolder.saveUser(userDTO);return true;}Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {UserHolder.removeUser();} }添加拦截器 Configuration public class WebMVCConfig implements WebMvcConfigurer {Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new LoginInterceptor()).excludePathPatterns(/shop/**,/voucher/**,/shop-type/**,/upload/**,/blog/hot,/user/code,/user/login);} }使用 Redis 存储验证码和用户信息 用 Session 存储验证码和用户信息的系统无法进行横向扩展因为多台 Tomcat 无法共享 Session。如果改用 Redis 存储就可以解决这个问题。 修改后的 UserService Log4j2 Service public class UserServiceImpl extends ServiceImplUserMapper, User implements IUserService {Autowiredprivate StringRedisTemplate stringRedisTemplate;private static final ObjectMapper OBJECT_MAPPER new ObjectMapper();Overridepublic Result sendCode(String phone, HttpSession session) {if (RegexUtils.isPhoneInvalid(phone)) {return Result.fail(不是合法的手机号);}String code RandomUtil.randomNumbers(6);stringRedisTemplate.opsForValue().set(LOGIN_CODE_KEY phone, code, LOGIN_CODE_TTL);// 发送短信log.debug(发送短信验证码{}, code);return Result.ok();}Overridepublic Result login(LoginFormDTO loginForm, HttpSession session) {// 验证手机号和验证码if (RegexUtils.isPhoneInvalid(loginForm.getPhone())) {return Result.fail(手机号不合法);}String code stringRedisTemplate.opsForValue().get(LOGIN_CODE_KEY loginForm.getPhone());if (code null || !code.equals(loginForm.getCode())) {return Result.fail(验证码不正确);}// 检查用户是否存在QueryWrapperUser qw new QueryWrapper();qw.eq(phone, loginForm.getPhone());User user this.baseMapper.selectOne(qw);if (user null) {user this.createUserByPhone(loginForm.getPhone());}// 将用户信息保存到 sessionString token UUID.randomUUID().toString(true);UserDTO userDTO new UserDTO();BeanUtils.copyProperties(user, userDTO);try {stringRedisTemplate.opsForValue().set(LOGIN_USER_KEY token,OBJECT_MAPPER.writeValueAsString(userDTO), LOGIN_USER_TTL);} catch (JsonProcessingException e) {e.printStackTrace();throw new RuntimeException(e);}return Result.ok(token);}private User createUserByPhone(String phone) {User user new User();user.setPhone(phone);user.setNickName(user_ RandomUtil.randomString(5));this.baseMapper.insert(user);return user;} }修改后的登录校验拦截器 public class LoginInterceptor implements HandlerInterceptor {private final StringRedisTemplate stringRedisTemplate;private static final ObjectMapper OBJECT_MAPPER new ObjectMapper();public LoginInterceptor(StringRedisTemplate stringRedisTemplate) {this.stringRedisTemplate stringRedisTemplate;}Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {// 从头信息获取 tokenString token request.getHeader(Authorization);if (ObjectUtils.isEmpty(token)) {// 缺少 tokenresponse.setStatus(401);return false;}// 从 Redis 获取用户信息String jsonUser this.stringRedisTemplate.opsForValue().get(LOGIN_USER_KEY token);UserDTO userDTO OBJECT_MAPPER.readValue(jsonUser, UserDTO.class);if (userDTO null) {response.setStatus(401);return false;}// 将用户信息保存到 ThreadLocalUserHolder.saveUser(userDTO);// 刷新 token 有效期stringRedisTemplate.expire(LOGIN_USER_KEY token, LOGIN_USER_TTL);return true;}Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {UserHolder.removeUser();} }还需要添加一个更新用户信息有效期的拦截器 public class RefreshTokenInterceptor implements HandlerInterceptor {private StringRedisTemplate stringRedisTemplate;public RefreshTokenInterceptor(StringRedisTemplate stringRedisTemplate) {this.stringRedisTemplate stringRedisTemplate;}Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {// 如果请求头中有 token且 redis 中有 token 相关的用户信息刷新其有效期String token request.getHeader(Authorization);if (ObjectUtils.isEmpty(token)) {return true;}if (Boolean.TRUE.equals(stringRedisTemplate.hasKey(LOGIN_USER_KEY token))) {stringRedisTemplate.expire(LOGIN_USER_KEY token, LOGIN_USER_TTL);}return true;} }添加这个新的拦截器并且确保其位于登录验证拦截器之前 Configuration public class WebMVCConfig implements WebMvcConfigurer {Autowiredprivate StringRedisTemplate stringRedisTemplate;Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new RefreshTokenInterceptor(stringRedisTemplate)).addPathPatterns(/**);registry.addInterceptor(new LoginInterceptor(stringRedisTemplate)).excludePathPatterns(/shop/**,/voucher/**,/shop-type/**,/upload/**,/blog/hot,/user/code,/user/login);} }商户查询 缓存 对商户类型查询使用 Redis 缓存以提高查询效率 Service public class ShopTypeServiceImpl extends ServiceImplShopTypeMapper, ShopType implements IShopTypeService {Autowiredprivate StringRedisTemplate stringRedisTemplate;Overridepublic Result queryTypeList() {String jsonTypeList stringRedisTemplate.opsForValue().get(CACHE_TYPE_LIST_KEY);if (!StringUtils.isEmpty(jsonTypeList)) {ListShopType typeList JSONUtil.toList(jsonTypeList, ShopType.class);return Result.ok(typeList);}ListShopType typeList this.query().orderByAsc(sort).list();if (!typeList.isEmpty()){stringRedisTemplate.opsForValue().set(CACHE_TYPE_LIST_KEY, JSONUtil.toJsonStr(typeList), CACHE_TYPE_LIST_TTL);}return Result.ok(typeList);} }对商户详情使用缓存 Service public class ShopServiceImpl extends ServiceImplShopMapper, Shop implements IShopService {private static final ObjectMapper OBJECT_MAPPER new ObjectMapper();Autowiredprivate StringRedisTemplate stringRedisTemplate;Overridepublic Result queryById(Long id) {// 先从 Redis 中查询String jsonShop stringRedisTemplate.opsForValue().get(CACHE_SHOP_KEY id);if (!StringUtils.isEmpty(jsonShop)) {Shop shop JSONUtil.toBean(jsonShop, Shop.class);return Result.ok(shop);}// Redis 中没有从数据库查Shop shop this.getById(id);if (shop ! null) {jsonShop JSONUtil.toJsonStr(shop);stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY id, jsonShop, CACHE_SHOP_TTL);}return Result.ok(shop);} }缓存更新策略 在编辑商户信息时将对应的缓存删除 Override public Result update(Shop shop) {if (shop.getId() null) {return Result.fail(商户id不能为空);}// 更新商户信息this.updateById(shop);// 删除缓存stringRedisTemplate.delete(CACHE_SHOP_KEY shop.getId());return Result.ok(); }缓存穿透 缓存穿透指如果请求的数据在缓存和数据库中都不存在就不会生成缓存数据每次请求都不会使用缓存会对数据库造成压力。 可以通过缓存空对象的方式解决缓存穿透问题。 在查询商铺信息时缓存空对象 Override public Result queryById(Long id) {// 先从 Redis 中查询String jsonShop stringRedisTemplate.opsForValue().get(CACHE_SHOP_KEY id);if (!StringUtils.isEmpty(jsonShop)) {Shop shop JSONUtil.toBean(jsonShop, Shop.class);return Result.ok(shop);}// Redis 中没有从数据库查Shop shop this.getById(id);if (shop ! null) {jsonShop JSONUtil.toJsonStr(shop);stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY id, jsonShop, CACHE_SHOP_TTL);return Result.ok(shop);} else {// 缓存空对象到缓存中stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY id, , CACHE_NULL_TTL);return Result.fail(店铺不存在);} }在这里缓存中的空对象用空字符串代替并且将缓存存活时间设置为一个较短的值比如说2分钟。 在从缓存中查询到空对象时返回商铺不存在 Override public Result queryById(Long id) {// 先从 Redis 中查询String jsonShop stringRedisTemplate.opsForValue().get(CACHE_SHOP_KEY id);if (!StringUtils.isEmpty(jsonShop)) {Shop shop JSONUtil.toBean(jsonShop, Shop.class);return Result.ok(shop);}// 如果从缓存中查询到空对象表示商铺不存在if (.equals(jsonShop)) {return Result.fail(商铺不存在);}// ... }缓存击穿 缓存击穿问题也叫热点Key问题就是一个被高并发访问并且缓存重建业务较复杂的key突然失效了无数的请求访问会在瞬间给数据库带来巨大的冲击。 常见的解决方案有两种 互斥锁逻辑过期 可以利用 Redis 做互斥锁来解决缓存击穿问题 Override public Result queryById(Long id) {// return queryWithCachePenetration(id);return queryWithCacheBreakdown(id); }/*** 用 Redis 创建互斥锁** param name 锁名称* return 成功/失败*/ private boolean lock(String name) {Boolean result stringRedisTemplate.opsForValue().setIfAbsent(name, 1, Duration.ofSeconds(10));return BooleanUtil.isTrue(result); }/*** 删除 Redis 互斥锁** param name 锁名称*/ private void unlock(String name) {stringRedisTemplate.delete(name); }/*** 查询店铺信息-缓存击穿** param id* return*/ private Result queryWithCacheBreakdown(Long id) {// 先查询是否存在缓存String jsonShop stringRedisTemplate.opsForValue().get(CACHE_SHOP_KEY id);if (!StringUtils.isEmpty(jsonShop)) {Shop shop JSONUtil.toBean(jsonShop, Shop.class);return Result.ok(shop);}// 如果从缓存中查询到空对象表示商铺不存在if (.equals(jsonShop)) {return Result.fail(商铺不存在);}// 缓存不存在尝试获取锁并创建缓存final String lockName lock:shop: id;try {if (!lock(lockName)){// 获取互斥锁失败休眠一段时间后重试Thread.sleep(50);return queryWithCacheBreakdown(id);}// 获取互斥锁成功创建缓存// 模拟长时间才能创建缓存Thread.sleep(100);Shop shop this.getById(id);if (shop ! null) {jsonShop JSONUtil.toJsonStr(shop);stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY id, jsonShop, CACHE_SHOP_TTL);return Result.ok(shop);} else {// 缓存空对象到缓存中stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY id, , CACHE_NULL_TTL);return Result.fail(店铺不存在);}} catch (InterruptedException e) {throw new RuntimeException(e);} finally {// 释放锁unlock(lockName);} }下面是用逻辑过期解决缓存击穿问题的方式。 首先需要将热点数据的缓存提前写入 Redis缓存预热 public class ShopServiceImpl extends ServiceImplShopMapper, Shop implements IShopService {/*** 创建店铺缓存** param id 店铺id* param duration 缓存有效时长*/public void saveShopCache(Long id, Duration duration) {Shop shop getById(id);RedisCacheShop redisCache new RedisCache();redisCache.setExpire(LocalDateTime.now().plus(duration));redisCache.setData(shop);stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY id, JSONUtil.toJsonStr(redisCache));}// ... }SpringBootTest class HmDianPingApplicationTests {Autowiredprivate ShopServiceImpl shopService;Testpublic void testSaveShopCache(){shopService.saveShopCache(1L, Duration.ofSeconds(1));}}Data public class RedisCacheT {private LocalDateTime expire; //逻辑过期时间private T data; // 数据 }Redis 中的缓存信息包含两部分过期时间和具体信息。大致如下 {data: {area: 大关,openHours: 10:00-22:00,sold: 4215,// ...},expire: 1708258021725 }且其 TTL 是-1也就是永不过期。 具体的缓存读取和重建逻辑 /*** 用逻辑过期解决缓存击穿问题** return*/ private Result queryWithLogicalExpiration(Long id) {//检查缓存是否存在String jsonShop stringRedisTemplate.opsForValue().get(CACHE_SHOP_KEY id);if (StringUtils.isEmpty(jsonShop)) {// 缓存不存在return Result.fail(店铺不存在);}// 缓存存在检查是否过期RedisCacheShop redisCache JSONUtil.toBean(jsonShop, new TypeReferenceRedisCacheShop() {}, true);if (redisCache.getExpire().isBefore(LocalDateTime.now())) {// 如果过期尝试获取互斥锁final String LOCK_NAME LOCK_SHOP_KEY id;if (lock(LOCK_NAME)) {// 获取互斥锁后单独启动线程更新缓存CACHE_UPDATE_ES.execute(() - {try {// 模拟缓存重建的延迟Thread.sleep(200);saveShopCache(id, Duration.ofSeconds(1));} catch (Exception e) {throw new RuntimeException(e);} finally {unlock(LOCK_NAME);}});}}// 无论是否过期返回缓存对象中的信息return Result.ok(redisCache.getData()); }封装 Redis 缓存工具类 可以对对 Redis 缓存相关逻辑进行封装可以避免在业务代码中重复编写相关逻辑。封装后分别对应以下方法 设置缓存数据TTL设置缓存数据逻辑过期时间从缓存获取数据用空对象解决缓存穿透问题从缓存获取数据用互斥锁解决缓存击穿问题从缓存获取数据用逻辑过期解决缓存击穿问题 工具类的完整代码可以参考这里。 本文的完整示例代码可以从这里获取。 参考资料 黑马程序员Redis入门到实战教程
http://www.zqtcl.cn/news/684037/

相关文章:

  • 科技网站域名大型网站开发团队
  • 温岭建设规划局网站注册新公司网上怎么核名
  • dede网站移动端怎么做golang 网站开发 开源
  • 织梦网站导航固定沈阳男科医院在线咨询免费
  • 四川华鸿建设有限公司网站网站建设需求文案
  • 汕头东莞网站建设怎么制作微信小程序app
  • 网站建设预算方案模板怎么做网络推广网站
  • 顺义网站开发wordpress内容主题模板下载
  • 永康做网站的化工网站模板免费下载
  • 潍坊高密网站建设如何做网页推广
  • 杭州房产网站建设wordpress 替换谷歌字体
  • 陕西省建设工程质量安全监督总站网站开发公司与施工单位工程造价鉴定报告
  • 嘉兴网站建设企业阿里巴巴国际贸易网站
  • 临沂手机网站建设珠海手机网站建设价格
  • 56m做图片视频的网站是什么软件开发和大数据哪个前景好
  • 郑州网站建设哪家做快消品的网站
  • 太原做网站费用东莞it外包
  • 深圳网站关键词优化公司集团网站建
  • 网站建设项目合同传奇手游网站
  • 如何学习网站建设app申请付费网站
  • 微网站开发平台案例重庆网站设计哪家公司好
  • 快递空包网站建设网站的首页怎么做的
  • 青海手机网站建设北京网站建设推荐华网天下
  • 网站网站建设公司孩子学编程网上课程哪家好
  • 跨境电商网站建设方案书江门网页制作
  • 门户网站建设定做如何使用域名访问网站
  • 做网站后台运营这个工作怎么样建设网站销售
  • 两学一做网上答题网站做网站域名是赠送的吗
  • 江苏住房城乡建设厅网站WordPress上传Excel
  • 广州淘宝网站建设济南高新区网站建设