Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 526|回复: 0

查询数据过多页面反应慢引入缓存解决方案(Redis、H2)

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-5-13 14:49:03 | 显示全部楼层 |阅读模式
     
    问题:原系统查询接口不支持分页也不可能加入分页支持,导致Ajax查询数据过多,返回数据达到2W多条记录时响应已经极慢,查询功能不要求数据实时性,页面反应速度极慢、体验不好;经排查是由于数据量过大导致写回页面的时候慢,实现是直接Servlet输出流写到页面上(output.write(buffer, 0, b));
     
    需求变更:加快页面相应速度,页面要有分页功能,可以改变原接口
    在这种情况下有两种解决方案:
                          1、前端js分页
                          2、加入第三方缓存部件(内存数据库等)
     
    两种解决方案的缺点:
        第一种方案的问题,虽然解决了分页问题但由于查询的数据量比较大,查询返回页面的数据达几万条有几兆甚至十多兆数据页面反应还是过慢,如果在前端使用js进行分页那查询反应过慢的问题还是没法决解,页面负担过重体验还是差;
        第二种方案问题,相对来说会比第一种复杂点,编码量会增多,如是处理实时查询较为复杂但也不是不可以,要考虑清理缓存数据已保证数据的有效性。
       
          经过权衡利弊,决定抛弃第一种方案,决定引入内存数据库已解决问题。
    内存数据库现在也有不少种类选择如:redis、H2、HSQLDB等等,在这三个内存数据中redis和其他两个还是不一样的,redis是C语言实现的轻量的基于key-value的nosql数据库,而H2与HSQLDB都是用java实现的轻量关系型数据库;所以我想在nosql数据库和关系型数据库中各选一个,实现此功能最终看性能如何决定使用哪个。 H2相比HSQLDB有web操作界面,H2比HSQLDB更新也比较频繁,所以这里选择H2与redis。
     
        一、redis缓存数据实现分页功能。
             redis实现不太友好的地方,由于redis是C语言实现的所以redis无法实现嵌入Java代码中启动redis服务器(我没找到方案),需要另外启        动redis服务器,为了实现分页还需要再启动redis视乎有点过了;
            思路:使用redis的zset(Sorted-Sets)有序集和数据类型来存储数据,SeeesionID+数据类型当作key,score为接口查询来的顺序,value为拼装好的单条json数据,可以根据Session失效或设置redis key的失效时间来清除数据。实现代码如下:
              /**
                 * 处理分页,查出数据填充到redis,然后再从redis中分页查询,每次查询都覆盖redis中数据
      * @param rtList
      * @param callback
      * @param response
      * @throws IOException
      * @linx
      * @Date 2014-11-10
                */
     public void processPager(List<CSTRTData> rtList, String callback,HttpServletRequest request,
       HttpServletResponse response,String top,String skip) throws IOException {
      Jedis jedis = new Jedis("127.0.0.1");
      OutputStream output = null;
      Set<String> setValues=new HashSet<String>();
      String dataList = "";
      String keyName=request.getSession().getId()+"troubleCode";
      if(top==null){
           top="1";
      }
      if(skip==null){
          skip="0";
      }
      if (rtList.size() > 0) {
       for (int x = 0; x < rtList.size(); x++) {
            JSONObject temp = JSONObject.fromObject(rtList.get(x));
            long recvTime = temp.getLong("recvTime");
            long obdTime = temp.getLong("obdTime");
            temp.remove("recvTime");
            temp.put("recvTime", DateUtil.formatYYYYMMDDHHMMSS(recvTime));
            temp.remove("obdTime");
            temp.put("obdTime", DateUtil.formatYYYYMMDDHHMMSS(obdTime));
            jedis.zadd(keyName, x, temp.toString());
                   
       }
       
       int start=Integer.valueOf(skip); //开始行号,第一页开始行号为0, 第二页起为上一页skip+页大小
       int end = Integer.valueOf(top)+Integer.valueOf(skip)-1; //结束行号 top=页大小, 结束行号=skip+top-1
         //根据start、end 序号从小到大 从redis中读取出数据
          setValues = jedis.zrange(keyName, start, end);
      //    Set<String> setValues2 = jedis.zrevrange("hackers", 0, -1);  
      //遍历拼装json
       for (Iterator iter = setValues.iterator(); iter.hasNext();) {
            if (iter.hasNext()) {
               dataList += (String) iter.next() + ",";
            } else {
               dataList += (String) iter.next();
            }
       }
      }else{
          jedis.del(keyName);
      }
      //拼装页面最终需要的json串
      dataList = callback + "({" + "\"d\" : {" + "\"results\" : [" + dataList
        + "]," + "\"__count\" : \"" + rtList.size() + "\"" + "}"
        + "})";
     // 转成输入流
     
      InputStream input = new ByteArrayInputStream(dataList.getBytes());
      output = response.getOutputStream();
      int b = 0;
      byte[] buffer = new byte[1024];
    //通过输出流写到页面
      while ((b = input.read(buffer)) != -1) {
       output.write(buffer, 0, b);
       output.flush();
      }
      if (output != null)
       output.close();
      if (input != null)
          input.close();
     }
       
     
      二、H2 缓存数据实现分页功能。
           H2有个好处就是可以嵌入在Java代码中启动H2数据库,由于它也是关系型数据库所以使用起来和我们常用的数据库没有什么区别,使用它的内存模式,你不用担心他的性能问题。
            思路:在H2中添加张表,然后把每条数据都转成JSON,然后按顺序存入添加的表中,就可以使用我们经常使用的分页Sql语句进行查询,每天点第一页的时候都调用原接口把数据重新覆盖到H2的表中。可以根据Session失效清除表数据。实现代码如下:
     
    public class H2Opertion {
     
         public static void createTable(Connection conn) throws Exception {
               Statement stmt = conn.createStatement();
              stmt.executeUpdate("CREATE TABLE temp(ID INT PRIMARY KEY,NAME VARCHAR(2000));");
     
         }
     
         public static Connection open() throws Exception {
             Class.forName("org.h2.Driver");
             Connection conn = DriverManager.getConnection("jdbc:h2:./h2db/demo","sa", "");
             return conn;
          }
     
        public static void insertData(Connection conn, List<String> list)
         throws Exception {
             Statement stmt = conn.createStatement();
             for (int i = 0; i < list.size(); i++) {
                  stmt.executeUpdate("INSERT INTO temp VALUES(" + i + ", '"   + list.get(i) + "');");
             }
         }
     
         public static List query(Connection conn,int start,int end) throws Exception {
      Statement stmt = conn.createStatement();
    //分页查询
      ResultSet rs = stmt
        .executeQuery("select t2.* from (select rownum r,t1.* from temp t1 where rownum<="+end+") t2 where t2.r>"+start);
      while (rs.next()) {
           System.out.println(rs.getInt("ID") + "," + rs.getString("NAME"));
      }
      return null;
     
         }
     
     public static void deleteTable(Connection conn) throws Exception {
         Statement stmt = conn.createStatement();
         stmt.executeUpdate("delete temp");
     }
     
     
     public static void main(String[] args) {
      H2Server server = new H2Server();
      server.startServer();
      Connection conn = null;
      try {
           conn = H2Opertion.open();
     
       try {
            H2Opertion.createTable(conn); //创建表
       } catch (Exception e) {
             e.printStackTrace();
             H2Opertion.log.error("表存在", e);
       }
       H2Opertion.log.info("删除表数据");
       H2Opertion.deleteTable(conn); //删除表数据
      //模拟接口查询出来的数据
       List<String> listData = new ArrayList<String>();
       for (int i = 0; i < 30; i++) {
           listData.add("content" + i);
       } 
          //把数据插入到H2的表中
           H2Opertion.insertData(conn, listData);  

          //分页查询H2表中的数据,10-20条数据

           H2Opertion.query(conn,10,20); 
      } catch (Exception e) {
          H2Opertion.log.error(e);
      } finally {
       try {
          conn.close();
       } catch (SQLException e) {
            H2Opertion.log.error(e);
       }
      }
        }
     
    }

    文章首发地址:Solinx 

    http://www.solinx.co/archives/67

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-23 03:16 , Processed in 0.065821 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

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