基本概念
页面优化技术
- 页面缓存+URL缓存+对象缓存
由于并发瓶颈在数据库,想办法如何减少对数据库的访问,所以加若干缓存来提高,通过各种粒度的缓存,最大粒度页面缓存到最小粒度的对象级缓存。 - 页面静态化,前后端分离
都是纯的html,通过js或者ajax来请求服务器,如果做了静态化,浏览器可以把html缓存在客户端。 - 静态资源优化
JS/CSS压缩,减少流量。(压缩版的js,去掉多余的空格字符。区别于阅读版)
JS/CSS组合,减少连接数。(将多个JS和CSS的组合到一个请求里面去,一下子从服务端全部下载下来) - CDN优化
内容分发网络,就近访问。
缓存特征:
-
命中率
当某个请求能够通过访问缓存而得到响应时,称为缓存命中。
缓存命中率越高,缓存的利用率也就越高。 -
最大空间
缓存通常位于内存中,内存的空间通常比磁盘空间小的多,因此缓存的最大空间不可能非常大。
当缓存存放的数据量超过最大空间时,就需要淘汰部分数据来存放新到达的数据。
淘汰策略
-
FIFO(First In First Out):先进先出策略,在实时性的场景下,需要经常访问最新的数据,那么就可以使用 FIFO,使得最先进入的数据(最晚的数据)被淘汰。
-
LRU(Least Recently Used):最近最久未使用策略,优先淘汰最久未使用的数据,也就是上次被访问时间距离现在最久的数据。该策略可以保证内存中的数据都是热点数据,也就是经常被访问的数据,从而保证缓存命中率。
-
LFU(Least Frequently Used):最不经常使用策略,优先淘汰一段时间内使用次数最少的数据。
一般,页面缓存和URL缓存时间比较短,适合场景:变化不大的页面。如果分页,不会全部缓存,一般缓存前一两页。
我在这个项目里面使用页面缓存+URL缓存+对象缓存,就不再去尝试前后端分离了。
缓存的思路
- 每次查的时候先查redis
- redis没有再去查数据库
- 将查出来的存到redis里面。
无论是对象还是我们的html页面都可以使用这个思路进行处理。
商品缓存:
在商品的service文件中进行操作。
思路很清晰,查缓存->没有就查数据库->存缓存->返回
@Override
public List<SkGoods> listGoods() {
//先去查缓存
List<SkGoods> goodsList = redisService.get(GoodsKey.GoodsList, "", List.class);
if (goodsList == null) {
logger.info("没有GoodsList缓存");
//缓存查不到,查数据库
goodsList = skGoodsDao.listGoods();
//存入缓存
redisService.set(GoodsKey.GoodsList, "", goodsList);
}else{
logger.info("我取的是GoodsList缓存");
}
return goodsList;
}
页面缓存
- 取缓存 (缓存里面存的是html)
- 手动渲染模板
- 结果输出(直接输出html代码)
在商品的Controller文件中:
/**
* 商品列表页面,如果是页面缓存的话就是存的html源码文件
* 是
*/
@RequestMapping(value = "/goods/to_list", produces = "text/html")
@ResponseBody
public String list(HttpServletRequest request, HttpServletResponse response, Model model) {
//取页面缓存,取得是html的源码
String html = redisService.get(GoodsKey.GoodsListHtml, "", String.class);
//取到的话就直接返回
if (html != null) {
return html;
}
//取商品缓存
List<SkGoods> goodsList=skGoodsService.listGoods();
model.addAttribute("goodsList", goodsList);
WebContext context = new WebContext(request, response,
request.getServletContext(), request.getLocale(), model.asMap());
html = templateEngine.process("goods_list", context);
if (html != null) {
redisService.set(GoodsKey.GoodsListHtml, "", html);
}
return html;
}
关键代码:
WebContext context = new WebContext(request, response,request.getServletContext(), request.getLocale(), model.asMap());
html = templateEngine.process("goods_list", context);
我们想要将html页面实现缓存的话就需要进行手动渲染然后保存到缓存中。缓存中存的就是html代码。而html中的参数都是需要通过context
来进行包装的。因此我们需要自己定义一个context
并设置参数,然后再使用templateEngine
来进行手动渲染。
其中:goods_list
是我们htnl文件的名称,contex携带了数据。
剩下的就是简单的缓存操作了,不再赘述。