文章目录
-
- 1.Redis页面缓存
-
-
- 1.思路分析
- 2.首先记录一下目前访问商品列表页的QPS
-
- 1.线程组配置10000次请求
- 2.请求配置
- 3.开始压测
-
- [1.压测第一次 平均QPS为612](#1.压测第一次 平均QPS为612)
- [2.压测第二次 平均QPS为615](#2.压测第二次 平均QPS为615)
- [3.压测第三次 平均QPS为617](#3.压测第三次 平均QPS为617)
- 3.然后记录一下访问商品详情页的QPS
-
- 1.线程组配置10000次请求
- 2.请求配置
- 3.开始压测
-
- [1.压测第一次 平均QPS为633](#1.压测第一次 平均QPS为633)
- [2.压测第二次 平均QPS为642](#2.压测第二次 平均QPS为642)
- [3.压测第三次 平均QPS为641](#3.压测第三次 平均QPS为641)
- 4.商品列表页Redis缓存优化
-
- 1.GoodsController.java
- [2.启动报错 Error creating bean with name 'requestMappingHandlerMapping' defined in class path resource](#2.启动报错 Error creating bean with name 'requestMappingHandlerMapping' defined in class path resource)
- 3.发现是produces写错了,将冒号换成分号
- 4.重新启动测试
- 5.商品详情页Redis缓存优化
- 6.压力测试
-
- 2.生产环境的压力测试
-
-
- 1.首先将GoodsController.java的从db查询商品列表打开
- 2.部署上线
- [3.UserUtil.java 获取用户信息](#3.UserUtil.java 获取用户信息)
- 4.准备压测
- 5.使用redis缓存页面来优化并重新部署
- 6.再次压测,QPS为80,有所提升
- 7.关于Redis缓存页面与DB的数据同步问题
-
- 3.对象缓存问题解决
1.Redis页面缓存
1.思路分析
data:image/s3,"s3://crabby-images/e85fc/e85fcd138d989942eeed65a9e63bdc482277d6cf" alt=""
2.首先记录一下目前访问商品列表页的QPS
1.线程组配置10000次请求
data:image/s3,"s3://crabby-images/5bd4e/5bd4ec54d7b7d4ac815b23693b26a9011b7b3d18" alt=""
2.请求配置
data:image/s3,"s3://crabby-images/74b0f/74b0fce47a0eaa8cb84a66fd686d6001cc8e5712" alt=""
3.开始压测
1.压测第一次 平均QPS为612
data:image/s3,"s3://crabby-images/7db0e/7db0e8ae9669875df59e4430d63e6f4d58e1ab9c" alt=""
2.压测第二次 平均QPS为615
data:image/s3,"s3://crabby-images/71167/711672b9752855ca32a8632eb4d084aee5b726c7" alt=""
3.压测第三次 平均QPS为617
data:image/s3,"s3://crabby-images/c2512/c25127bc2da38bc239e547423b89d45e0cb0858c" alt=""
3.然后记录一下访问商品详情页的QPS
1.线程组配置10000次请求
data:image/s3,"s3://crabby-images/5bd4e/5bd4ec54d7b7d4ac815b23693b26a9011b7b3d18" alt=""
2.请求配置
data:image/s3,"s3://crabby-images/8ae2a/8ae2a508c742dc16339f763d1c2c09ec9b3f7752" alt=""
3.开始压测
1.压测第一次 平均QPS为633
data:image/s3,"s3://crabby-images/88d33/88d33941adda70b6760e8726fc084d55f2c42243" alt=""
2.压测第二次 平均QPS为642
data:image/s3,"s3://crabby-images/98939/98939b57d215c84832851a8ecbb5856c6f833b97" alt=""
3.压测第三次 平均QPS为641
data:image/s3,"s3://crabby-images/995e5/995e597f4540d94d7079f944aba1242539814590" alt=""
4.商品列表页Redis缓存优化
1.GoodsController.java
java
@Resource
GoodsService goodsService;
@Resource
private RedisTemplate redisTemplate;
@Resource
private ThymeleafViewResolver thymeleafViewResolver;
// 进入到商品首页-使用redis优化
@RequestMapping(value = "/toList", produces = "text/html;charset=UTF-8")
@ResponseBody
public String toList(Model model, User user, HttpServletRequest request, HttpServletResponse response) {
// 判断是否有用户信息
if (null == user) {
return "login";
}
// 先从redis中获取页面,如果有则直接返回
String html = (String) redisTemplate.opsForValue().get("goodsList");
if (StringUtils.hasText(html)) {
return html;
}
// 如果没有就从数据库中查询,然后存入redis中
// ------------------------------db查询商品列表------------------------------
// 查询商品列表
model.addAttribute("goodsList", goodsService.findGoodsVo());
// 将用户信息存入model中
model.addAttribute("user", user);
// ------------------------------db查询商品列表------------------------------
// 渲染页面
// 1.首先构建一个webContext对象,用来存放model
WebContext context = new WebContext(request, response, request.getServletContext(), request.getLocale(), model.asMap());
// 2.渲染页面
html = thymeleafViewResolver.getTemplateEngine().process("goodsList", context);
// 3.判断html是否为空,不为空则存入redis中,设置过期时间为60s
if (StringUtils.hasText(html)) {
redisTemplate.opsForValue().set("goodsList", html, 180, TimeUnit.SECONDS);
}
return html;
}
2.启动报错 Error creating bean with name 'requestMappingHandlerMapping' defined in class path resource
data:image/s3,"s3://crabby-images/32fad/32fadd468c62068165ba41b94f36f433d99974bc" alt=""
3.发现是produces写错了,将冒号换成分号
data:image/s3,"s3://crabby-images/68814/688140380b270878dca88fb8d8c54d6b669ea76f" alt=""
4.重新启动测试
1.登录后访问 http://localhost:9092/seckill/goods/toList
data:image/s3,"s3://crabby-images/b915c/b915ce5a9986d94b60a8fafb808a7daf1c345ab9" alt=""
2.此时如果在数据库中修改信息,在60s内是不会刷新的
data:image/s3,"s3://crabby-images/31020/310200fa4fe8731003283e4cd68a63434a212545" alt=""
data:image/s3,"s3://crabby-images/1de12/1de126c9204adb4581ff1e0aae2956ed887aca67" alt=""
5.商品详情页Redis缓存优化
1.GoodsController.java
java
// 进入到商品详情页
@RequestMapping(value = "/toDetail/{goodsId}", produces = "text/html;charset=UTF-8")
@ResponseBody
public String toDetail(Model model, User user, @PathVariable Long goodsId, HttpServletRequest request, HttpServletResponse response) {
// 判断是否有用户信息
if (null == user) {
return "login";
}
// 先从redis中获取页面,如果有则直接返回
String html = (String) redisTemplate.opsForValue().get("goodsDetail:" + goodsId);
// 如果有则直接返回
if (StringUtils.hasText(html)) {
return html;
}
// 如果没有就从数据库中查询,然后存入redis中
// ------------------------------db查询商品详情------------------------------
// 查询商品详情
GoodsVo goodsVoByGoodsId = goodsService.findGoodsVoByGoodsId(goodsId);
model.addAttribute("goods", goodsVoByGoodsId);
// secKillStatus:秒杀状态 0:未开始 1:进行中 2:已结束
// remainSeconds:秒杀剩余时间 >0:未开始 0:进行中 -1:已结束
// 获取该商品的秒杀开始时间和结束时间
long startAt = goodsVoByGoodsId.getStartDate().getTime();
long endAt = goodsVoByGoodsId.getEndDate().getTime();
long now = System.currentTimeMillis();
// 根据当前时间与秒杀开始时间和结束时间的比较,判断秒杀状态
int secKillStatus = 0;
int remainSeconds = 0;
if (now < startAt) {
// 秒杀未开始
secKillStatus = 0;
remainSeconds = (int) ((startAt - now) / 1000);
} else if (now > endAt) {
// 秒杀已结束
secKillStatus = 2;
remainSeconds = -1;
} else {
// 秒杀进行中
secKillStatus = 1;
remainSeconds = 0;
}
// 将秒杀状态和剩余时间存入model中,返回到前端
model.addAttribute("secKillStatus", secKillStatus);
model.addAttribute("remainSeconds", remainSeconds);
// 将用户信息存入model中,返回到前端
model.addAttribute("user", user);
// ------------------------------db查询商品详情------------------------------
// 渲染页面
// 1.首先构建一个webContext对象,用来存放model
WebContext context = new WebContext(request, response, request.getServletContext(), request.getLocale(), model.asMap());
// 2.渲染页面
html = thymeleafViewResolver.getTemplateEngine().process("goodsDetail", context);
// 3.判断html是否为空,不为空则存入redis中,设置过期时间为60s
if (StringUtils.hasText(html)) {
redisTemplate.opsForValue().set("goodsDetail:" + goodsId, html, 180, TimeUnit.SECONDS);
}
return html;
}
2.测试
1.登录后访问详情页
data:image/s3,"s3://crabby-images/ba71b/ba71b8d176f709b1f5fa54191a6252e039e88eef" alt=""
2.在Redis中也有了缓存
data:image/s3,"s3://crabby-images/b7ea7/b7ea73c823df83cdb8a22d320a0d29bcabb92f21" alt=""
6.压力测试
1.清空Redis
data:image/s3,"s3://crabby-images/e324b/e324b288ca727add33508f070ca412286b3eb691" alt=""
2.清空用户表
data:image/s3,"s3://crabby-images/1f1ea/1f1ea84989b1942f2b83f7b4c5cab57e5970855a" alt=""
3.启动应用,重新生成2000个用户
data:image/s3,"s3://crabby-images/c357d/c357d6d31a86a7a97df21e6c9a2e96f8fd850b47" alt=""
4.对访问列表页进行压测
1.发现平均QPS只有80,比直接走数据库还慢
data:image/s3,"s3://crabby-images/6db4f/6db4fb9e91a4eb46a32e943f42c47f5875cacdf8" alt=""
2.由于六台机器都开启了RDB和AOF的持久化策略,现在分别将其关闭,然后重启redis
data:image/s3,"s3://crabby-images/d8eb0/d8eb0c832fb4a32d1fc74b01377a1ccdb13dfd28" alt=""
3.再次压测,还是80
data:image/s3,"s3://crabby-images/967e9/967e98eb5705ff7be32e58baa399e9a89b315b8d" alt=""
4.那么就可能是网络原因了,因为服务器都在北京,所以将服务部署到生产环境然后再进行压力测试
2.生产环境的压力测试
1.首先将GoodsController.java的从db查询商品列表打开
data:image/s3,"s3://crabby-images/25330/253301d9bba6842e234013d060f5a6d4a39dd3e0" alt=""
2.部署上线
1.激活环境为prod
data:image/s3,"s3://crabby-images/4cfae/4cfae8c95e6b6c77f59a5d3b1d2e548de63029c5" alt=""
2.maven打包
data:image/s3,"s3://crabby-images/08041/0804190a40cc4af90eb49369f8495b1d5344516d" alt=""
3.上传到服务器然后重新启动
data:image/s3,"s3://crabby-images/05f38/05f382ce632260ae30f25a983e8b3f94528c947f" alt=""
data:image/s3,"s3://crabby-images/4ad2f/4ad2fdf38546dd8120bc3a5b10c8e192c7a810a8" alt=""
3.UserUtil.java 获取用户信息
1.修改环境变量
data:image/s3,"s3://crabby-images/7e771/7e771602ea3e18406b454212d7872a0d49d3a8da" alt=""
2.获取cookie
data:image/s3,"s3://crabby-images/83a62/83a62beeff6837d2dd575a41efb55b01dcad2789" alt=""
4.准备压测
1.http请求默认值
data:image/s3,"s3://crabby-images/ad7bf/ad7bf2b384ce05af5f60ab585a8de6749ad54863" alt=""
2.http请求信息
data:image/s3,"s3://crabby-images/d3f97/d3f9713dafb2203585d50a3a9bc525c8400ff184" alt=""
3.修改cookie的域
data:image/s3,"s3://crabby-images/7f001/7f00145e46572e473fefbece00777667232ee815" alt=""
4.开始压测5000次请求,QPS为55
data:image/s3,"s3://crabby-images/22441/22441f721641f5d9b5bb73a2b55a511d264e9816" alt=""
5.使用redis缓存页面来优化并重新部署
data:image/s3,"s3://crabby-images/9b2ec/9b2ecbf2986a9fc2706a44d4306a6555bf9bb3f0" alt=""
6.再次压测,QPS为80,有所提升
data:image/s3,"s3://crabby-images/81dd5/81dd5a5bf54af4efb059c8b2b47362ca47f81799" alt=""
7.关于Redis缓存页面与DB的数据同步问题
data:image/s3,"s3://crabby-images/ddc88/ddc882bbf74c3d5286db2293b0bbc14edf0072c7" alt=""
3.对象缓存问题解决
1.问题分析
在校验用户是否登录时,会根据cookie在Redis中查询用户信息,但是如果在DB中的用户信息更改了,那么就会发生数据不一致的问题
2.具体实现
1.UserService.java
java
/**
* 更新密码
* @param userTicket
* @param password
* @param request
* @param response
* @return
*/
public RespBean updatePassword(String userTicket, String password, HttpServletRequest request, HttpServletResponse response);
2.UserServiceImpl.java
java
@Override
public RespBean updatePassword(String userTicket, String password, HttpServletRequest request, HttpServletResponse response) {
// 根据ticket获取用户
User user = getUserByCookie(userTicket, request, response);
if (null == user) {
throw new GlobalException(RespBeanEnum.MOBILE_NOT_EXIST);
}
// 更新密码
user.setPassword(MD5Util.inputPassToDBPass(password, user.getSlat()));
int result = userMapper.updateById(user);
if (1 == result) {
// 删除redis中的用户信息
redisTemplate.delete("user:" + userTicket);
return RespBean.success();
}
return RespBean.error(RespBeanEnum.PASSWORD_UPDATE_FAIL);
}
3.UserController.java
java
// 更新密码
@RequestMapping("/updatePassword")
@ResponseBody
public RespBean updatePassword(String userTicket, String password, HttpServletRequest request, HttpServletResponse response) {
return userService.updatePassword(userTicket, password, request, response);
}
3.测试
1.登录一下,得到票据
data:image/s3,"s3://crabby-images/1d8cd/1d8cdd534fd0f407f25fb96e619b0118385fc5d8" alt=""
2.Redis中有该用户信息
data:image/s3,"s3://crabby-images/f8984/f8984988a44055d71649a19d55e4fe2ef85c81c4" alt=""
3.更新密码 http://localhost:9092/seckill/user/updatePassword?userTicket=4dfaea799a9b438ea96ef61f7da435e3\&password=666666
data:image/s3,"s3://crabby-images/c9340/c93406c0298cb3f8924a113b22279bbc531c26e2" alt=""
4.刷新Redis,用户信息被删除
data:image/s3,"s3://crabby-images/96b6f/96b6fbf28dfd8d007e37b5fc6d8b19a78c931cae" alt=""