设为首页 收藏本站
查看: 610|回复: 0

[经验分享] 小白的web优化之路 一、使用redis来缓存信息

[复制链接]

尚未签到

发表于 2018-11-4 07:05:48 | 显示全部楼层 |阅读模式
  作为一个web开发者,根据产品经理的需求来完成一个应用显然不是我们的唯一目标,很多时候,我们考虑的不仅仅是需求,而是超越需求,认识到项目上线后的性能瓶颈。很多web应用在上线后,都可能会碰到响应慢的问题,而从今天开始,我将以浅显的例子带领大家一步一步的优化web应用。
  先举个栗子:
  小白在做一个web列表页时,需求很简单,当用户访问第一个页的时候,服务器返回1-10条的文章简介数据,访问第2页的时候,服务器返回11-20条的文章简介数据。小白转念一想,这简单啊,每次前端给我一个页数就行,我直接从数据库中查出来,然后返回回去就行。哈哈,so easy!
  当小白把程序写好上线后,随着用户以及文章的增多,数据库的压力也越大,比如当文章数量达到一百万条时,每次分页查询都需要耗时数据库500ms左右,这时候小白就意识到程序需要优化了。
  这时候小白的老师来了,老师说,这些文章列表数据,在短时间中不会改变太大,为什么不把它缓存起来呢?这样每次都可以从内存中取出来,而不用每次都访问数据库了。
  即项目架构由最简单的

  加入了cache层:

  这样,可以极大的加快访问速度。
  就拿这个例子来说,你可以把每次获取的列表信心都缓存起来,为了在缓存的同时也能保持更新,你可以设置列表缓存为1分钟过期,这样这1分钟内多次访问速度会极大的提高。
  至于缓存,你可以使用Redis,Redis 是一个高性能的key-value数据库。可以对关系数据库起到很好的补充作用。
  小白在听了老师给出的建议后,马上着手开始搭建redis开发环境,并开始开发,他写了一个demo,这个demo中,使用缓存前每次访问的速度为400-500ms,使用缓存后的访问速度稳定在25-27ms,相当于17倍的速度提升!
  下面为主要的测试代码:
  [java] view plain copy

  •   package com.happyheng.controller;

  •   import com.alibaba.fastjson.JSON;
  •   import com.happyheng.dao.ArticleDao;
  •   import com.happyheng.model.Article;
  •   import org.springframework.beans.factory.annotation.Autowired;
  •   import org.springframework.util.StringUtils;
  •   import org.springframework.web.bind.annotation.RequestMapping;
  •   import org.springframework.web.bind.annotation.RestController;
  •   import redis.clients.jedis.Jedis;

  •   import java.util.List;

  •   /**
  •   *
  •   *
  •   * Created by happyheng on 17/5/6.
  •   */
  •   @RestController
  •   @RequestMapping("/test")
  •   public class TestController {


  •   private static final String KEY_CACHE_ARTICLE_LIST = "article_list";

  •   @Autowired
  •   private ArticleDao articleDao;


  •   @RequestMapping("/testSelectFromDb")
  •   public List testSelectFromDb() throws Exception {

  •   return articleDao.getArticleListFromdb();
  •   }

  •   @RequestMapping("/testSelectFromCache")
  •   public String testSelectFromCache() throws Exception {

  •   Jedis jedis = new Jedis("localhost");
  •   // 先从缓存中取出数据
  •   String articleListStr = jedis.get(KEY_CACHE_ARTICLE_LIST);
  •   if (!StringUtils.isEmpty(articleListStr)) {

  •   return articleListStr;
  •   } else {

  •   // 如果数据为空,那么从db中取出来,然后序列化后写入到cache中,并设置1分钟的过期时间
  •   List articleList = articleDao.getArticleListFromdb();
  •   String serializeArticleListStr = JSON.toJSONString(articleList);
  •   jedis.set(KEY_CACHE_ARTICLE_LIST, serializeArticleListStr);
  •   jedis.expire(KEY_CACHE_ARTICLE_LIST, 60);
  •   return serializeArticleListStr;
  •   }

  •   }


  •   }
  [java] view plain copy

  •   package com.happyheng.dao;

  •   import com.happyheng.model.Article;
  •   import org.springframework.stereotype.Service;

  •   import java.sql.*;
  •   import java.util.ArrayList;
  •   import java.util.List;

  •   /**
  •   *
  •   *
  •   * Created by happyheng on 17/5/7.
  •   */
  •   @Service
  •   public class ArticleDao {


  •   public List getArticleListFromdb() throws Exception{

  •   String sql = "SELECT * FROM article ORDER BY create_time DESC LIMIT 0, 10";

  •   Class.forName("com.mysql.jdbc.Driver");
  •   Connection mConnection = DriverManager.getConnection("jdbc:mysql://localhost:3306/optimize_db", "root", "mytestcon");
  •   Statement statement = mConnection.createStatement();
  •   ResultSet resultSet = statement.executeQuery(sql);

  •   try {
  •   List list = new ArrayList();
  •   //注意指针刚开始是-1位置,这行next()方法,会先判断下一个位置有没有,如果有,指向下一个位置。
  •   while (resultSet.next()) {
  •   Article article = new Article();
  •   article.setId(resultSet.getLong("id"));
  •   article.setTitle(resultSet.getString("title"));
  •   article.setContent(resultSet.getString("content"));
  •   article.setCreate_time(resultSet.getString("create_time"));
  •   list.add(article);
  •   }
  •   return list;
  •   } catch (Exception e) {
  •   e.printStackTrace();
  •   } finally {
  •   try {
  •   resultSet.close();
  •   statement.close();
  •   mConnection.close();
  •   } catch (SQLException e) {
  •   e.printStackTrace();
  •   }

  •   }

  •   return null;
  •   }
  •   }
  可以看到,当访问 testSelectFromDb 时,直接从数据库中获取数据,访问 testSelectFromCache 时,先从cache中获取数据,如果没有,从数据库中获取数据,然后写入到cache中,下面为两种访问方法的访问时间对比:
  直接访问db:

  访问cache:



运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-630397-1-1.html 上篇帖子: Redis数据类型之字符串String-12873930 下篇帖子: Spring-data-redis操作redis知识总结
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表