首页 > 代码库 > OA项目11:部门列表树状显示功能及其他代码优化
OA项目11:部门列表树状显示功能及其他代码优化
首注:本学习教程为传智播客汤阳光讲师所公布的免费OA项目视频我的文字版实践笔记,本人用此来加强巩固自己开发知识,如有网友转载,请注明。谢谢。
一 使用递归使部门列表树状显示:
1.写一个工具类,实现通过顶级部门查询所有,具体如下:
1 package cn.clear.oa.util; 2 3 import java.util.ArrayList; 4 import java.util.Collection; 5 import java.util.List; 6 7 import cn.clear.oa.domain.Department; 8 9 public class DepartmentUtil {10 //遍历部门树,把所有部门遍历出来以树状显示11 public static List<Department> getAllDepartments(List<Department> topList) {12 List<Department> departments = new ArrayList<Department>();13 walkDepartmentTreeList(topList,"┣ ",departments);14 return departments;15 }16 17 private static void walkDepartmentTreeList(Collection<Department> topList,String prefix, List<Department> departments){18 19 for (Department top : topList) {20 Department copy = new Department();//创建副本,不要使用在Session的原对象21 //遍历顶点22 copy.setId(top.getId());23 copy.setName(prefix + top.getName());24 departments.add(copy);25 //遍历子树 26 walkDepartmentTreeList(top.getChildren()," "+prefix,departments);27 }28 }29 }
2.因为部门列表在新建页面和修改页面存在,所有修改新建页面和修改页面的action,如下:
1 public String addUI() throws Exception {2 // 准备departmentList数据3 List<Department> topList = departmentService.findTopList();4 List<Department> departmentList = DepartmentUtil5 .getAllDepartments(topList);6 // 放在值栈中的map中7 ActionContext.getContext().put("departmentList", departmentList);8 return "saveUI";9 }
1 public String editUI() throws Exception { 2 // 准备departmentList数据 3 List<Department> topList = departmentService.findTopList(); 4 List<Department> departmentList = DepartmentUtil 5 .getAllDepartments(topList); 6 // 放在值栈中的map中 7 ActionContext.getContext().put("departmentList", departmentList); 8 9 Department department = departmentService.findById(model.getId());10 // 将对象放在栈顶11 ActionContext.getContext().getValueStack().push(department);12 // 回显上级部门13 if (department.getParent() != null) {14 parentId = department.getParent().getId();15 }16 return "saveUI";17 }
3.发现树形列表的顺序混乱(即刷新会改变顺序),这是因为在hibernate映射文件中使用了set集合,是无序的,要使得按顺序排列,那么修改映射文件,在查询子部门列表集合时增加一个order-by="id ASC"标签,这个标签的值应该是sql的排序语句,具体如下:
二 对所有jsp页面进行优化:
将所有公共的jsp导入代码抽取到一个新的页面,之后只要引用该新页面即可,不用再在每个页面进行多次导入操作了。
新页面commons.jspf:
1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>2 <%@ taglib prefix="s" uri="/struts-tags"%> 3 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />4 <script language="javascript" src="${pageContext.request.contextPath}/script/jquery.js"></script>5 <script language="javascript" src="${pageContext.request.contextPath}/script/pageCommon.js" charset="utf-8"></script>6 <script language="javascript" src="${pageContext.request.contextPath}/script/PageUtils.js" charset="utf-8"></script>7 <link type="text/css" rel="stylesheet" href="${pageContext.request.contextPath}/style/blue/pageCommon.css" />8 <script type="text/javascript">9 </script>
其他页面的和commons.jspf代码相同的全部删除,然后在页面中设置一个包含标签,将commons页面引入即可:
<%@ include file="/WEB-INF/jsp/public/commons.jspf" %>
三 抽取后台action的相同代码,放到BaseAction.java中,在使用时只要继承该类即可:
1.建立BaseAction.java:
1 package cn.clear.oa.base; 2 3 import java.lang.reflect.ParameterizedType; 4 5 import javax.annotation.Resource; 6 7 import cn.clear.oa.service.DepartmentService; 8 import cn.clear.oa.service.RoleService; 9 10 import com.opensymphony.xwork2.ActionSupport;11 import com.opensymphony.xwork2.ModelDriven;12 13 @SuppressWarnings("unchecked")14 public abstract class BaseAction<T> extends ActionSupport implements15 ModelDriven<T> {16 17 /**18 * 19 */20 private static final long serialVersionUID = 1L;21 //=======================ModelDriven的支持==========================22 protected T model;23 24 public BaseAction(){25 26 try {27 //通过反射获取model的真实类型28 ParameterizedType pt = (ParameterizedType) this.getClass().getGenericSuperclass();29 30 Class<T> clazz = (Class<T>) pt.getActualTypeArguments()[0];31 //通过反射创建model的实例32 model = clazz.newInstance();33 } catch (Exception e) {34 // TODO Auto-generated catch block35 throw new RuntimeException(e);36 }37 }38 39 public T getModel() {40 // TODO Auto-generated method stub41 return model;42 }43 44 45 //=========================Service的声明============================46 @Resource47 protected RoleService roleService;48 @Resource49 protected DepartmentService departmentService;50 }
2.在具体action中将与BaseAction.java相同的代码去除,指定具体action只继承BaseAction即可,这样在action就可以专注自己的代码了。
四 将Dao层和Service层代码结合在Service层,减少两层间的重用代码:
1.将Dao层所有代码(包括实现类)全部删除;
2.此时因为Dao层已经消亡,所以将BaseDao.java及BaseDaoImpl.java名字改为DaoSupport.java及DaoSupportImpl.java,以告知这是Dao数据库操作支持;
3.将Service层的接口类直接实现DaoSupport.java,将Service层的实现类继承DaoSupportImpl.java,并将所有重复的业务代码删除即可。
以下就是修改后的整体结构图:
五 修改以上后测试发现无法修改和删除实体,究其原因是因为Service的实现类开启事务方法是使用@Transactional注解,但是Service的实现类继承自DaoSupportImpl.java,而该类是无法使用子类的事务,所以解决的方法是在DaoSupportImpl.java中也使用该注解开启事务,此注解可以继承,即子类可以字节使用父类开启的事务,所以在Service的实现类中可以将该注解移除。修改完之后开启服务器测试,Ok。
OA项目11:部门列表树状显示功能及其他代码优化