1、AJAX+J2EE 开发组织机构管理系统(1)2006-02-20 10:40 作者: 黎宇 出处:天极 ( 11 ) 砖 ( 12 ) 好 评论 ( 2 ) 条 进入论坛 关 键 词:AJAX J2EE阅读提示:本文件通过开发一个组织机构管理小系统全面介绍了 Ajax 开发的各个细节。通过与 J2ee 的结合来实现三层分布式开发的层次划分,后台与前端的调用。 一、 概述AJAX 是今年初才问世的新技术,是 Asynchronous JavaScript and XML 的缩写。它是一组开发 Web 应用程序的技术,它使浏览器可以为用户提供更为自然的浏览体验。每当需要更新时,客户端 Web 页
2、面的修改是异步的和逐步增加的。这样,AJAX 在提交 Web 页面内容时大大提高了用户界面的速度。在基于 AJAX 的应用程序中没有必要长时间等待整个页面的刷新。页面中需要更新的那部分才进行更改,如果可能的话,更新是在本地完成的,并且是异步的。J2ee 是一种用来开发分布式系统的体系结构。它主要是用 Java 类开发业务实体。通过 JSP来连接应用服务器。本文开发一个组织机构管理小系统,通过这个实例来介绍如何用 Ajax 开发 WEB 应用程序。本系统具有增加、修改、删除组织机构的功能。同时给机构分配人员,能增加、修改、删除人员。二、 界面设计树结构是大多软件系统中常采用的结构形式。由于树型结
3、构层次分明、上下级关系清楚、且展开收缩表达信息方便、界面也较美观,所以是大家热衷于用此结构。组织机构管理是一般软件基本具有的。组织机构是指公司的组织结构。集团公司可包括分公子公司,公司下面又有科室。员工归属于所在的公司。系统运行后的界面如下:orgManager.htm 是组织机构管理的主页面。WEB 应用程序界面设计是非常重要的。如何布局、么样组织可直接体现一个人的设计水平。组织机构主要包括树结构、组织机构编辑、人员编辑等三大块,如何分成三块呢,然而一般树型结构的窗体常先二块,树型结构独占一块,另一块又分成上下二部分,上面是机构编码,下面是人员编码。固可以把页面划分成如下图形式:组织编码区
4、2树结构区 1 人员管理区 3 显然我们是通过表来实现。这是一个二行二列的表,且第一、二行的左边列合并单元格。代码如下: TABLE border=“1“ width=“100%“ height=“100%“TRTD rowspan=“2“FONT face=“宋体“/FONT/TDTD/TD/TRTRTD/TD/TR/TABLE我们在 1 区(单元格)上加上一个 DIV,因为 DIV 可以动态地滚动,并且可以插入其它控件。DIV 的 id 为“divTree“,且风格设置为溢出时自动滚动,宽与高都为 100%,及满区域。代码如下:div id=“divTree“ style=“width:1
5、00%; height:100%;background-color:#f5f5f5;border :1px solid Silver;overflow:auto;“/div我们在 2 区(单元格)上也加上一个 DIV,在 DIV 里再插入一个表格。表格上放下控件,这很简单,就不详细说了。我们在 3 区(单元格)上加上一个 DIV。此 DIV 的 id 为“ divContent “,且风格设置为竖直溢出时自动滚动,宽与高都为 100%,及满区域,此 DIV 用来装载人员信息;在 DIV 里再插入一个表格, 此 table 的 id 为“ tbList “,是用来输入、显示人员作息,同时在此表中
6、插入一些如 checkbox 、text、select 等控件。说明,表的第二列是用来放人员唯一编号的,不显示。代码如下:div id=“divContent“ style=“height:100%; overflow-y:auto;“ width=“100%“table id=“tbList“ border=“1“ width=“100%“tr seqNo=“1“tdtable border=“1“ width=“100%“trtd width=“5%“input type=“checkbox“ value=“on“/input /tdtd width=“0%“ style=“display
7、:none“ input type=“text“ size=“20“/input/tdtd width=“40%“input type=“text“ size=“20“/input/tdtd width=“25%“select size=“1“ name=“D1“option value=“0“男/optionoption selected=“true“ value=“1“女/option/select/td/tr/table /td/tr/table/div 三、 前端页面的主要编码1. 树的实现在 WEB 上实现树结构,同样我们是通过 Ajax 来实现的。树上可以显示自定义的图标,可以插入
8、、删除、结点。并且结点可任意移动。这里我们不重点讲树的实现技术,我们已经封装好了,你只要按要求去改动就是了。 1) 键接树型文件在head与/head之间键接我们的与树有关的文件, 代码如下:link rel=“STYLESHEET“ type=“text/css“ href=“css/dhtmlXTree.css“script src=“js/dhtmlXCommon.js“/scriptscript src=“js/dhtmlXTree.js“/script 2) 装载方法在页面的文档打开时装载自定义方法, preLoadImages 方法实现树控件的图标定义,doOnLoad 实现树控件
9、的图标定义代码如下:body onload=“preLoadImages();doOnLoad();“3) 编写方法/doOnLoad 实现装载并显示树。设置树属性等。function doOnLoad()OrgTree=new dhtmlXTreeObject(document.getElementById(divTree),“100%“,“100%“,0); /dhtmlXTreeObject 是树对象,通过新建对象,指定树显示的 DIV 可定义树。OrgTree.setImagePath(“imgs/“);/设置树的图片所在位置OrgTree.setDragHandler();/设置树结
10、点拖动OrgTree.enableDragAndDrop(true) /设置树结点是否可拖动OrgTree.setDragHandler(myDragHandler); /设置树结点拖动时所执行的方法OrgTree.setOnClickHandler(mySelectHandler); /设置树单击时所执行的方法/OrgTree.setXMLAutoLoading(“Org.jsp“);/装载树结点数据。数据来源如 Org.jsp 所返回的 XML 格式的字符串,数据是动态装载,且当展开时才装载。OrgTree.loadXML(“root.xml?0“);/装载树结点数据。数据来源 root.
11、xml 文件,并且从 xml 文件的 ID号为 0 处读取数据。/OrgTree.loadXML(“Org.jsp“);/装载树结点数据。数据来源如 Org.jsp 所返回的 XML 格式的字符串,并且是一次性全部装载数据。 /preLoadImages 方法实现树控件的图标定义function preLoadImages()var imSrcAr = new Array(“line1.gif“,“line2.gif“,“line3.gif“,“line4.gif“,“minus2.gif“,“minus3.gif“,“minus4.gif“,“plus2.gif“,“plus3.gif“,“
12、plus4.gif“,“book.gif“,“books_open.gif“,“books_close.gif“,“magazine_open.gif“,“magazine_close.gif“,“tombs.gif“,“tombs_mag.gif“,“book_titel.gif“)var imAr = new Array(0);for(var i=0;iimSrcAr.length;i+)imArimAr.length = new Image();imArimAr.length-1.src = “imgs/“+imSrcAri. 组织管理的实现组织可以增加、删除、编辑。同时当选择树结点时应
13、该把组织显示出来供编辑,查看。为了实现这些功能,你只要按要求去改动就是了。1) 全局变量的定义许多地方我们要用到一些公共变量,我们在script与/script之间定义全局变量, 代码如下:var OrgTree = null; /组织树 Domvar nextSeq = 0;/人员管理的顺序号(流水号)var personDom;/人员 Domvar CurrNodeId;/当前结点 Id 2) 初始化当页面打开时我们要控件好那部分该显示,那部分要隐藏。且对全局变量的赋值等,组织类型装载。在页面的文档打开时装载自定义方法 init(), init 方法实现初始化。body onload=“i
14、nit();“init 方法实现如下:function init() /定义 personDom 为一个 XMLDOM对象personDom= new ActiveXObject(Microsoft.XMLDOM);personDom.async = false; /定义 stylesheet 为一个 XMLDOM对象,且 stylesheet 为 personDom 确定显示风格 stylesheet = new ActiveXObject(Microsoft.XMLDOM);stylesheet.async = false;stylesheet.load(“addOrgPerson.xsl
15、“); /装载 stylesheet 的风格定义文件/装载组织类型数据var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=GetOrgType“, false);xmlhttp.setRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send(); retXml=xmlhttp.responseText;/ alert(retXml); /把组织类型插入下拉列表控件中 var
16、 OrgDoc = new ActiveXObject(Microsoft.XMLDOM);OrgDoc.async = false;OrgDoc.loadXML(retXml);var root = OrgDoc.documentElement;oNodeList = root.childNodes;txtType.options.length =oNodeList.length;for (var i=0; ioNodeList.length; i+) Item = oNodeList.item(i); var OrgTypeId=Item.childNodes(0).text;var Or
17、gTypeName=Item.childNodes(1).text;txtType.optionsi.value=OrgTypeId;txtType.optionsi.text=OrgTypeName;/ txtType.options0. 3) 编写树拖动及选择结点的方法/ myDragHandler 实现树结点拖动时重新指定父子关系。function myDragHandler(idFrom,idTo)var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=moveOrgx
18、mlhttp.setRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send();retXml=xmlhttp.OrgponseText;return true;/ mySelectHandler 实现选择树结点对系统的控制,同时显示组织信息及该组织下的人员。function mySelectHandler(id) tbOrg.style.display=“block“;divOrgMemo.style.display=“none“;divOrgInfo.style.display=“none
19、“; if(id=1)divOrgMemo.style.display=“block“;div1.style.display=“none“;div2.style.display=“none“;div3.style.display=“none“;divContent.style.display=“none“;div5.style.display=“none“; elsedivOrgInfo.style.display=“block“; div1.style.display=“block“;div2.style.display=“block“;div3.style.display=“block“;
20、divContent.style.display=“block“;div5.style.display=“block“; CurrNodeId=id;/装载组织信息并显示在编码和名称的文本控件上。 loadOrg(id);/装载某组织下人员信息 var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=GetPersonxmlhttp.setRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xml
21、http.send();retXml=xmlhttp.responseText; personDom.loadXML (retXml);/给人员信息的每行加上序号 for(var i=0; ipersonDom.documentElement.childNodes.length; i+)personDom.documentElement.childNodesi.setAttribute(“seqNo“, nextSeq);nextSeq+; /人员信息显示在 divContent 上面divContent.innerHTML = personDom.transformNode(styleshe
22、et); ;/装载组织信息并显示在编码和名称的文本控件上。function loadOrg(OrgId) if(OrgId = null)OrgId = OrgTree.getSelectedItemId();if(OrgId = “)tbOrg.style.display = “none“;return;var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=loadOrgxmlhttp.setRequestHeader(“Content-Type“, “applicatio
23、n/x-www-form-urlencoded“);xmlhttp.send();retXml=xmlhttp.responseText;var OrgDoc = new ActiveXObject(Microsoft.XMLDOM);OrgDoc.async = false;OrgDoc.loadXML(retXml);if(OrgId != 1)txtCode.value = OrgDoc.selectSingleNode(“/OrgCode“).text;txtName.value = OrgDoc.selectSingleNode(“/OrgName“).text;tbOrg.styl
24、e.display = “block“; 4) 建立组织组织建立主要是通过调用 XMLHTTP 对象来实现。我们主要学会如何调用 XMLHTTP。组织建立应该在后台实现,把组织信息插入数据库中。这里我们通过 JSP 来实现。我们的Org.jsp 文件中有个 createOrg 方法,该方法传递一个父 ID。function createOrg(parentOrgId)var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=createOrgxmlhttp.setRequestH
25、eader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send();retXml=xmlhttp.responseText;var orgId = (new Number(retXml).toString();return orgId;5) 删除组织组织删除同样是调用 Org.jsp 文件中的 deleteOrg 方法来实现,该方法传递所删除的结点ID。function deleteOrg()var OrgId = OrgTree.getSelectedItemId();var xmlhttp = new Acti
26、veXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=deleteOrgxmlhttp.setRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send(); 6) 编辑组织组织修改是调用 Org.jsp 文件中的 modifyOrg 方法来实现,该方法传递所修改的结点 ID。同时修改的数据通过自定义的 XML 格式的字符串传送,这时通过 send 字符串来实现。修改前数据一律要验证其合法性,并提示错误信息。functi
27、on modifyOrg()if(OrgTree.getSelectedItemId() = “)return “N“;if(txtCode.value = “)alert(“请输入编码!“);return “N“;if(txtName.value = “)alert(“请输入名称!“);return“N“; var OrgId = OrgTree.getSelectedItemId();var OrgKind; /alert(txtType.optionstxtType.selectedIndex.value)var strModify = “?xml version=1.0 encodin
28、g=gb2312?“ +“data“ + “OrgCode!CDATA“ + txtCode.value + “/OrgCode“ +“OrgName!CDATA“ + txtName.value + “/OrgName“ +“OrgKind!CDATA“ + txtType.optionstxtType.selectedIndex.value+ “/OrgKind“ +“/data“;var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=modifyOrgxmlhttp.s
29、etRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send(strModify);OrgTree.setItemText(OrgTree.getSelectedItemId(),txtName.value);3. 人员管理的实现人员可以增加、删除、编辑。同时当选择树结点时应该把人员显示出来供编辑、查看1) 增加人员人员增加实现的原理是在 personDom 中加入结点 peorsone,该结点相当于表的一行,设置属性。同时在 peorsone 中不继地加入其它结点,代表数据库的字段,且必须与
30、 XLT 文件的标号同名。这些结点相当该行的列。最后在表中插入一行,行上插入一列,并显示之。function addPerson() var seqNo = nextSeq;nextSeq+; var peorsonNode = personDom.createNode(“1“, “peorsone“,“);peorsonNode.setAttribute(“isNew“, “Y“);peorsonNode.setAttribute(“isDelete“, “N“);peorsonNode.setAttribute(“seqNo“, seqNo);personDom.documentEleme
31、nt.appendChild(peorsonNode);var PersonId= personDom.createNode(“1“, “personId“, “);peorsonNode.appendChild(PersonId);var personCode= personDom.createNode(“1“, “personCode“, “);peorsonNode.appendChild(personCode);var PersonName= personDom.createNode(“1“, “personName“, “);peorsonNode.appendChild(Perso
32、nName);var Sex= personDom.createNode(“1“, “sex“, “);peorsonNode.appendChild(Sex); var tr = tbList.insertRow(tbList.rows.length);tr.setAttribute(“seqNo“, seqNo);var td = tr.insertCell(0);td.innerHTML = peorsonNode.transformNode(stylesheet); 2) 删除人员人员删除同样是调用 Org.jsp 文件中的 deletePerson 方法来实现,该方法传递所删除的人员
33、 ID。如何确定人员 ID 是通过读取隐藏的 ID,并扫描整个表,看那些被选中。这里我们要注意是提供多项选择的。function deletePerson() for(var i=0; itbList.rows.length; i+)var row=tbList.rowsi.cells0.children0.rows0;if(row.cells0.children0.checked)var personId=row.cells1.children0.value; if(personId0) var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“)
34、;xmlhttp.open(“POST“,“Org.jsp?mode=deletePersonxmlhttp.setRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send(); tbList.deleteRow(i);i-; 3) 编辑人员人员修改我们要判定哪些行被修改了。刚增加但没保存的行应该是新增而不是修改的。function save() if( modifyOrg()=“N“)return;for(var i=0; itbList.rows.length; i+) var row=t
35、bList.rowsi.cells0.children0.rows0;var personId=row.cells1.children0.value;var seqNo = tbList.rowsi.getAttribute(“seqNo“);var staffNode = personDom.selectSingleNode(“/peorsoneseqNo=“ + seqNo + “);var personCode=row.cells2.children0.value; var personName=row.cells3.children0.value; var sex=row.cells4
36、.children0.value; /alert(staffN;ode ); if(staffNode.getAttribute(“isNew“) = “Y“) createPerson(CurrNodeId,personCode,personName,sex); else var strXML = “?xml version=1.0 encoding=gb2312?“ +“data“ + “personCode!CDATA“ + personCode+ “/personCode“ +“personName!CDATA“ + personName + “/personName“ +“sex!C
37、DATA“ + sex+ “/sex“ +“personId!CDATA“ + personId+ “/personId“ +“/data“; /alert(strXML ); var xmlhttp = new ActiveXObject(“Microsoft.XMLHTTP“);xmlhttp.open(“POST“,“Org.jsp?mode=modifyPerson“, false);xmlhttp.setRequestHeader(“Content-Type“, “application/x-www-form-urlencoded“);xmlhttp.send(strXML ); 四
38、、 XML 与 XSL 文件设计XML 是种可扩展的标记语言,它具有开放的、可扩展的、可自描述的语言结构,它已经成为网上数据和文档传输的标准。XSLT 的目的是将信息内容与 Web 显示分离,HTML 通过按抽象概念(如段落、重点和编号列表)定义显示来实现设备独立性。XSLT 用来具体显示控件,设置控件风格。Ajax 主要使用 XML 和 XSLT 进行数据交换与处理。1. 树信息的 XML 文件(见 root.xml 文件) XML 是标记语言,元素必须成对出现。树结构中以 tree 为根结点,以 item 为结点体,属性 text 指出结点所显示的文本,id 指出唯一的所标识号。?xml
39、version=1.0 encoding=gb2312?tree id=“0“item child=“1“ text=“组织“ id=“1“ /item /tree这文件并不是必要的,只是为了系统能独立运行才加的。事实如果连接了后台数据是不需要的。只要吧 OrgTree.loadXML(“root.xml?0“)改为 OrgTree.loadXML(“Org.jsp“)就可以了。2. 人员信息 XML 文件(见 peorson.xml 文件)说明!CDATA可在任何显示任何格式的文本,文本中可插入其它任何字符。这文件也不是必要的。3. 人员信息展现的 xsl 文件(见 addOrgPerson
40、.xsl 文件)xsl 文件同样是 XML 格式文件。所以一律遵守 XML 标准。下面对主要的行讲解:?xml version=“1.0“ encoding=“gb2312“?/这是定义 xml 文件的首行。用来指明版本及字符集xsl:stylesheet xmlns:xsl=“http:/www.w3.org/TR/WD-xsl“ language=“JavaScript“/这里定义了 stylesheet 元素。并指出其国际命名的组织及语言。 xsl:template match=“/“xsl:apply-templates select=“peorsones“/xsl:template/
41、上面是匹配的规则。“/“表示从根结开始去匹配。匹配到下面的 peorsones 标记。这是正则表达式有关的学问。我们只要理解就可以。xsl:template match=“peorsones“/当匹配上 peorsones 时所要做的事情。table id=“tbList“ border=“1“ width=“100%“/定义一个 id 为“tbList 的表格。此表格是显示在 WEB 上的xsl:for-each select=“peorsone“/循环匹配 peorsonetr/定义 tbList 表格的一行,并在行上增加一个叫 seqNo 的属性名,值为匹配到的seqNo(序号)xsl:
42、attribute name=“seqNo“xsl:value-of select=“seqNo“/xsl:attributetd/定义行上的一列,列又去匹配xsl:apply-templates select=“.“/td/tr/xsl:for-each /table/xsl:templatexsl:template match=“peorsone“table border=“1“ width=“100%“tr/定义宽为 5%的一列,在该列上插入一个 checkbox 控件td width=“5%“input type=“checkbox“ value=“on“ size=“10“/inpu
43、t/td/定义一个不显示的列,在该列上插入一个 text 控件,text 的值为匹配到的personId(人员 Id)td style=“display:none“input type=“text“ size=“25“xsl:attribute name=“value“xsl:value-of select=“personId“/xsl:attribute/input /tdtd width=“30%“ input type=“text“ size=“20“ xsl:attribute name=“value“xsl:value-of select=“personCode“/xsl:attri
44、bute /input /td td width=“40%“ input type=“text“ size=“40“xsl:attributename=“value“xsl:value-of select=“personName“/xsl:attribute /input/td/定义一个 width 为 28%的列,在该列上插入一个下拉列表 select 控件,select 的值如果匹配到为 0 时则为“男“,1 时则为“女“td width=“28%“select size=“1“option value=“0“xsl:if test=“.sex=0“xsl:attribute name=“
45、selected“true/xsl:attribute/xsl:if男/optionoption value=“1“xsl:if test=“.sex=1“xsl:attribute name=“selected“true/xsl:attribute/xsl:if 女/option/select/td/定义一列,在该列上插入一个 button 控件,onclick 事件为自定义的方法,该方法传递当前单击的按纽td width=“*“ button onclick=“openPersonRolePage(this)“ style=“width: 36; height: 21“角色/button/
46、td/tr/table/xsl:template/xsl:stylesheet五、 数据接口的实现(见 Org.jpg 文件)Org.JSP 文件用来在服务器上运行 Java 的类与前台 web 页之间架起一座桥。取到中间件的接口作用。这里分析部分代码:% page contentType=“text/html; charset=GBK“ % page import=“java.sql.*“ % page import=“javax.naming.*“ % page import=“javax.sql.*“ % page import=“tool.*“ % page import=“orgNe
47、w.*“ % page import=“org.w3c.dom.*“ %/上面主要是引用一些 java 类%try/request.setCharacterEncoding(“GBK“);Document doc = XmlTool.createDocumentFromRequest(request);/建立 web 面文档请求的文档对象Connection conn = ConnTool.getConnectionFromPool();/获取请求的方法名String mode=request.getParameter(“mode“);/out.println(“ccc“);/如果方法中没有其它参数则读取组织树数据if(mode = null) /* int OrgId = Integer.parseInt(request.getParameter(“id“);String str = orgManager.getChildOrg(OrgId, conn);out.println(str); */String str = orgManager.getTree(conn);/out.println(str);out.println(str); else if(mode.equals(“createOrg“