java树型结构的数据展现设计
在做一个需求管理的页面时,需求的展现是不限层级树型结构,需求下还可以分拆任务,页面要展现的字段有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树型结构的数据展现设计的更多相关文章
- dzzoffice的树型结构用户管理设计
在DzzOffice1.1的开发中,针对用户使用群体重新设计了,机构.部门.用户管理应用. 传统OA,企业相关程序,一般是设置机构-设置部门-设置职位-添加用户这样的步骤.每个步骤分为不同的管理界面. ...
- JSP中的一个树型结构
看方力勋的javaWeb,采用左右值来表示树型结构(就是俺门的多级分类)表结构 页面代码 <%@ page language="java" import="java ...
- 20-Ubuntu-文件和目录命令-查看目录树型结构-tree
tree 以树状图列出当前目录下的文件目录结构 选项 含义 -d 只显示当前目录的子目录树型结构 显示当前目录的子目录和文件树型结构 例: 1.查看文档目录下的子目录和文件树型结构 2.查看文档目 ...
- 分享使用NPOI导出Excel树状结构的数据,如部门用户菜单权限
大家都知道使用NPOI导出Excel格式数据 很简单,网上一搜,到处都有示例代码. 因为工作的关系,经常会有处理各种数据库数据的场景,其中处理Excel 数据导出,以备客户人员确认数据,场景很常见. ...
- Java实现树形结构的数据转Json格式
在项目中难免会用到树形结构,毕竟这是一种常用的组织架构.楼主这里整理了两个实现的版本,可以直接拿来使用,非常方便. 楼主没有单独建项目,直接在以前的一个Demo上实现的.第一种,看下面代码: pack ...
- java实现树型结构样式
import javax.swing.*; import javax.swing.event.*; import javax.swing.tree.*; public class Root exten ...
- java树状结构之二叉树
参考:http://blog.csdn.net/zhangerqing/article/details/8822476 前面已经提到过树和二叉树的相关概念内容,下面主要来介绍下关于二叉树的创建,遍历, ...
- Delphi实现树型结构
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...
- Delphi实现树型结构具体实例
unit Unit1;interfaceuses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, ...
随机推荐
- nginx-1.5.10 之mips编译到RT5350
编译nginx-1.5.10一般须要下面库的支持:pcre,zlib,openssl 此次编译nginx-1.5.10使用的库版本号分别为pcre-8.34:openssl-1.0.0l:zlib-1 ...
- POJ2773 Happy 2006【容斥原理】
题目链接: http://poj.org/problem?id=2773 题目大意: 给你两个整数N和K.找到第k个与N互素的数(互素的数从小到大排列).当中 (1 <= m <= 100 ...
- Linux watchdog 6300esb
基本原理: Linux 自带了一个 watchdog 的实现,用于监视系统的执行,包含一个内核 watchdog module 和一个用户空间的 watchdog 程序.内核 watchdog ...
- Android解析程序包时出现问题
Android用户下载我们wcc应用时,偶尔会出现“解析程序包出现问题”的的现象,以下是逐步排查的相关经验: 1. 首先确保这个包本身没有问题. 检测方法:其他手机采用同样的下载方式再下载一次. 解决 ...
- POJ 2299 Ultra-QuickSort(线段树+离散化)
题目地址:POJ 2299 这题以前用归并排序做过.线段树加上离散化也能够做.一般线段树的话会超时. 这题的数字最大到10^10次方,显然太大,可是能够利用下标,下标总共仅仅有50w.能够从数字大的開 ...
- Objective-C 内存管理retain和release
OC使用引用计数来管理内存,每个继承NSObject的对象,内部都维护了一个引用计数器retainCount.当对象创建时(调用alloc或者new)引用计数器会+1, 手动调用retain()方法能 ...
- eventkeyboardmouse
http://www.javascriptkit.com/jsref/eventkeyboardmouse.shtml
- MySQL - MyCat 实现读写分离
前言 MyCat是一个彻底开源的,面向企业应用开发的大数据库集群,支持事务.ACID.可以替代MySQL的加强版数据库.其功能有可以视为MySQL集群的企业级数据库,用来替代昂贵的Oracle集群.融 ...
- bzoj 1629: [Usaco2007 Demo]Cow Acrobats【贪心+排序】
仿佛学到了贪心的新姿势-- 考虑相邻两头牛,交换它们对其他牛不产生影响,所以如果交换这两头牛能使这两头牛之间的最大值变小,则交换 #include<iostream> #include&l ...
- Linux 搭建Discuz论坛
title: Linux 搭建Discuz论坛 Welcome to Fofade's Blog! 这里是Linux 搭建论坛的一些命令记录 命令摘记: 下载文件:Discuz 安装环境:PHP Ap ...