在做一个需求管理的页面时,需求的展现是不限层级树型结构,需求下还可以分拆任务,页面要展现的字段有20多个,而且需求采用通用表单设计,db采用大宽表存储,有一百多个字段。目前数据量不大,第一版采用普通的同步加载数据方式,页面加载速度慢,慢的原因主要是代码逻辑没有做到最优,不如存在重复查询的问题,组织树型结构的时候也存在冗余的查询。加上页面渲染出全部数据,使得整个页面打开速度很慢。

针对以上问题,同事做了第二版优化,主要采用ajax加载页面,滚动条采用代码控制,拖动滚动条的位置计算要显示的数据,然后从数据库取出相应的数据。第二版主要存在问题是每次滚动鼠标都会发起请求加载数据,并且每次请求的数据都会覆盖已经加载的数据,在加载过程中使用遮罩禁止重复滚动加载数据。以上问题使得用户体验很差,所以我临时接手进行了第二次优化。

思路:

1.针对每个项目的需求和任务,在第一次访问的时候,将当前访问的项目的全部需求和任务树型结构组织好并缓存起来,缓存过期时间控制在2-4小时(依数据量和内存大小灵活设置),注意采用indexMap.put(i++, row);的形式,这样方便后期分屏加载。row的定义为Map<String, Object> row ;//Object可存需求和任务,通过key区分。

2.获取查询条件,如果为空,则从数据库取最后一次保存的查询条件,否则保存最新查询条件。

3.通过遍历indexMap,统计满足条件的需求和任务数量。

4.返回框架页面,并发送ajax请求真实数据。

5.对请求分页处理,分页大小控制在一次请求50条,太大页面渲染时间慢,太小又需较多次加载。遍历indexMap,取出满足分页条件和查询条件的数据,返回给客户端。

6.页面滚动到末尾,自动发起ajax加载后面的数据。

通过瀑布流的形式,解决了树型结构的分页问题,不用一次全部取出数据,使用缓存,提高加载速度。通过第三次优化。用户体验得到了较大提升。

注意点:使用原生ajax返回50条数据拼接到原有数据末尾的时候,ie容易内存溢出。

部分关键代码:

/**

判断最后一行是否进入可视区域

obj:有滚动条的容器box

lastRow:最后一行

**/

function see(obj, lastRow){

  if(obj && lastRow){

    var seeHeight = document.documentElement.clientHeight || document.body.clientHeight;//可视高度

    var winScroll = obj.scrollTop;//获取滚动条当前位置

    var lastSee = obj.scrollHeight;//最后一行

    return lastSww< (seeHeight + winScroll)?true:false;

  }else{

    return true;

  }

}

/**

  将ajax数据拼接到页面末尾

**/

function append(parent, text){

  if(typeof text ==='string'){

    var temp = document.createElement('div');

    temp.innerHTML = text;

    var frag = document.createDocumentFragment();

    while(temp.firstChild){

      frag.appendChild(temp.firstChild);

    }

    parent.appendChild(frag);

    delete temp;

    delete frag;

    CollectGarbage();

  }

}

/**

  组织treeIndex

**/

... ...

int i = 0;

for(ProjectDemand mode: items){

  Map<String, Object> row = new HashMap<String, Object>();

  if(attachMap.get(mode.getId())!=null){

    mode.setArrachmetCount(attachMap.get(mode.getId()));

  }

  row.put.("demand", mode);

  indexMap.put(i++, row);

  if(ScrumTaskMap.get(mode.getId())!=null){

    List<Task> tasklist = scrumTaskMap.get(mode.getId());

    for(Task task : tasklist){

      Map<String, object> rowTask  = new HashMap<String, Object>();

      rowTask.put("task", task);

      indexMap.put(i++, rowTask);

    }

  }

}

/**

统计需求和任务数量

**/

private String projectDemandStatistics(Map indexMap, Map<String,String> filter, Map<integer, FormBase> showDemandAll, Map<Integer, Task> showTaskAll ){

  int demandAllCount=0;

  int leafCount=0;

  int taskCount = 0;

  int allCount = indexMap.size();

  FormBase demand = null;

  FormBase tempDemand = null;

  Task task = null;

  Map<String, String>SignMap = SZSEUtils.getItemsSign(indexMap);

  String showDemandFlag = TypeUtils.nullToString(filter.get("showDemandFlag")); 

  Map<Integer, FormBase> iteredDemand = new HashMap();//已遍历需求

  

  for(int i=0; i<allCount;i++){

    Map<String, Object> row = (Map<String,  Object>) indexMap.get(i);

    if(row==null) break;

    if(row.get("demand")!=null){
      demand = (FormBase) row.get("demand");

      iteredDemand.put(demand.getId(), demand);

      if(isShowDemand(filter, demand)){\

        if(showDemandAll!=null) showDemandAll.p;ut(demand.getId(), demand);//在这里记录要显示的需求map,供任务显示与否判断时用

        demandAllCount++;

        if(!"+".equals(signMap.get(demand.getId()+""))){

          leafCount++;

        }

        //拾遗父需求

        tempDemand = demand;

        while(showDemandAll !=null

          && SZSEUtils.isNotEmpty(tempDemand.getNum01())

          && !"0".equals(tempDemand.getNum01())

          &&showDemandAll.get(TypeUtils.nullToInt(tempDemand.getNum01()))==null){

          tempDemand = itereDemand.get(Typeutils.nullToInt(tempDemand.getNum01()));

          showDemandAll.put(tempDemand.getId(), tempDemand);

          demandAllCount++;

          if(!"+".equals(signMap.get(tempDemand.getId()+""))){

            leafCount++;

          }

        }

      }

    }else if("true".equals(showDemandFlag)){//显示任务

      task = (Task) row.get("taskId");

      if(isShowTask(filter, task)){

        int parentID = TypeUtils.nullToInt(task.getStr07());

        if(showDemandAll!=null && showDemandAll.get(parentID)!=null){//如果需求不显示,则任务也不显示

          showTaskAll.put(task.getId9), task);

          taskCount++;

        }

      }

    }

  }

  String countMessage = "共"+ demandAllCount +"个需求,共"+leafCount+"个叶子需求";

  if("true".equals(showDemandFlag)) countMessage +="; 共"+taskCount+"个任务";

  countMessage+=".";

  return countMessage;

}

java树型结构的数据展现设计的更多相关文章

  1. dzzoffice的树型结构用户管理设计

    在DzzOffice1.1的开发中,针对用户使用群体重新设计了,机构.部门.用户管理应用. 传统OA,企业相关程序,一般是设置机构-设置部门-设置职位-添加用户这样的步骤.每个步骤分为不同的管理界面. ...

  2. JSP中的一个树型结构

    看方力勋的javaWeb,采用左右值来表示树型结构(就是俺门的多级分类)表结构 页面代码 <%@ page language="java" import="java ...

  3. 20-Ubuntu-文件和目录命令-查看目录树型结构-tree

    tree 以树状图列出当前目录下的文件目录结构 选项 含义 -d 只显示当前目录的子目录树型结构   显示当前目录的子目录和文件树型结构 例: 1.查看文档目录下的子目录和文件树型结构 2.查看文档目 ...

  4. 分享使用NPOI导出Excel树状结构的数据,如部门用户菜单权限

    大家都知道使用NPOI导出Excel格式数据 很简单,网上一搜,到处都有示例代码. 因为工作的关系,经常会有处理各种数据库数据的场景,其中处理Excel 数据导出,以备客户人员确认数据,场景很常见. ...

  5. Java实现树形结构的数据转Json格式

    在项目中难免会用到树形结构,毕竟这是一种常用的组织架构.楼主这里整理了两个实现的版本,可以直接拿来使用,非常方便. 楼主没有单独建项目,直接在以前的一个Demo上实现的.第一种,看下面代码: pack ...

  6. java实现树型结构样式

    import javax.swing.*; import javax.swing.event.*; import javax.swing.tree.*; public class Root exten ...

  7. java树状结构之二叉树

    参考:http://blog.csdn.net/zhangerqing/article/details/8822476 前面已经提到过树和二叉树的相关概念内容,下面主要来介绍下关于二叉树的创建,遍历, ...

  8. Delphi实现树型结构

    unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...

  9. Delphi实现树型结构具体实例

    unit Unit1;interfaceuses  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, ...

随机推荐

  1. cakephp2.3.8中何为component

    大胆假设,小心求证         记得这句话是以前读高三的时候,一个数学老师(圆头,有点胖胖的老师,不是很记得),经常挂在嘴边的一句话, 对于我们不理解,或者说是无法确定的东西,我们不妨先大胆地去假 ...

  2. 【SSH 基础】浅谈Hibernate--入门篇

    Hibernate是什么 Hibernate是一个轻量级的ORMapping框架 ORMapping原理(Object Relational Mapping)就是把对象里面的数据和数据库里面的数据,依 ...

  3. python 字符编码处理问题总结 彻底击碎乱码!

    Python中常常遇到这种字符编码问题,尤其在处理网页源代码时(特别是爬虫中): UnicodeDecodeError: 'XXX' codec can't decode bytes in posit ...

  4. initial ram disk

    1 什么是initial ram disk 它就是一个做好了的文件系统,其存储空间是ram.在kernel启动的第一个阶段,会被mount成根文件系统. 2 为什么需要initial ram disk ...

  5. python的一些常用函数

    1 filter(function, iterable) 等价于(item for item in iterable if function(item)) 就是说,filter会遍历iterable中 ...

  6. spring 之 IOC 依赖注入详解

    当我们对一个javaBean进行实例化时,在原本的情况下我们会选择新建一个接口,然后进行实例化,为了进一步降低耦合度我们还会使用工厂模式进行封装. 例: 当我们想要去造,Chinese.America ...

  7. go09---defer

    package main /* defer 类似其它语言中的析构函数,在函数体执行结束后 按照调用顺序的相反顺序逐个执行,先进后出, 即使函数发生严重错误也会执行,资源清理文件关闭, 支持匿名函数的调 ...

  8. 学习MAP 地图好地址

    http://www.cnblogs.com/beniao/archive/2010/01/13/1646446.html Bēniaǒ成长笔记在IT江湖里我不是一名老手,我只是一名普通的程序员,愿意 ...

  9. POJ1751 Tree 树分治

    分析:每次找重心可以发现最多n层,每层复杂度是O(nlogn) 总体时间复杂度是O(nlog^2n) #include <cstdio> #include <cstring> ...

  10. mac系统下的常用命令

    这是我日常在mac下记录的一些常用终端命令: 1 java 2 javac 3 exit 4 /Users/lianxumac/Desktop/apktool1.5.2/反编译 ; exit; 5 / ...