Chinaunix首页 | 论坛 | 博客
  • 博客访问: 30465030
  • 博文数量: 708
  • 博客积分: 12163
  • 博客等级: 上将
  • 技术积分: 8240
  • 用 户 组: 普通用户
  • 注册时间: 2007-12-04 20:59
文章分类

全部博文(708)

分类: 系统运维

2008-03-17 11:48:58

现在到处都有这方面的教程,我重点说一下我自己搞的一个框架。 

特点: 
支持Form的无闪提交(方法有点笨) 
支持MVC框架,即支持传统网页架构 
多线程并发请求(要语言支持线程) 
动态加载文件,只加载有用的!处理了Ajax框架臃肿的JS文件问题。 
采用no table的全div + css布局 

a. 获得XMLHTTPRequest对象,网上到处都找得到了,不多说: 

function newXMLHttpRequest() { 
var xmlreq = false; 
if (window.XMLHttpRequest) { 
xmlreq = new XMLHttpRequest(); 
} else if (window.ActiveXObject) { 
try { 
xmlreq = new ActiveXObject("Msxml2.XMLHTTP"); 
} catch (e1) { 
try { 
xmlreq = new ActiveXObject("Microsoft.XMLHTTP"); 
} catch (e2) { 



return xmlreq; 

这里提供一个通用的支持多浏览器的方法。 

b.提出异步请求 



//这里用Bcandy作为方法名是为了感谢一个对我来说很重要的人,她一直在支持我 
function Bcandy(Tid,url,parm,js) { 
if(url == ""){ 
return; 

//这是一个加载信息提示框,也可以不要! 
document.getElementById("load").style.visibility = "visible"; 
//加载相应页面的JS文件 
if(js != null){ 
//加载JS文件 
LoadJS(js); 

// 获取一个XMLHttpRequest实例 
var req = newXMLHttpRequest(); 
// 设置用来从请求对象接收回调通知的句柄函数 
var handlerFunction = getReadyStateHandler(req,Tid); 
req.onreadystatechange = handlerFunction; 
// 第三个参数表示请求是异步的 
req.open("POST", url, true); 
// 指示请求体包含form数据 
req.setRequestHeader("Content-Type", 
"application/x-www-form-urlencoded"); 
// 发送参数 
req.send(parm); 


function getReadyStateHandler(req,Tid) { 
// 返回一个监听XMLHttpRequest实例的匿名函数 
return function () { 
// 如果请求的状态是“完成” 
if (req.readyState == 4) { 
// 成功接收了服务器响应 
if (req.status == 200) { 
//下面一句是重点,这里显示了返回信息的内容部分,也可以加以修改。进行其它处理 
document.getElementById(Tid).innerHTML = req.responseText; 
document.getElementById(Tid).style.visibility = "visible"; 
//这一句是实现加载信息提示框的隐藏,也可以不要。 
document.getElementById("load").style.visibility = "hidden"; 
} else { 
// 有HTTP问题发生 
document.getElementById("load").style.visibility = "hidden"; 
alert("HTTP error: "+req.status); 






//动态加载JS文件 
function LoadJS(file){ 
var head = document.getElementsByTagName(’HEAD’).item(0); 
var script = document.createElement(’SCRIPT’); 
script.src = file; 
script.type = "text/javascript"; 
head.appendChild(script); 

这就是基本的框架了,因为使用了request.responseText;所以,可以直接请求一个页面jsp,servlet但在使用Struts框架的请求时要进行特殊处理,因为Form不支持异步请求。建议在这些页面上不要加入标签,就像.net里的asxm文件!而且在使用Struts框架时有点要注意的是,Mapping对象直接返回null就可以了,因为我们会在下面讲到并发多线程。来处理这个问题的。 
总的来看,有点像是积木搭建起来的。这样方便文件的修改和扩展,互相之间并不影响,而且,实现了代码和标签分离。在进行传统页面改版时,也不用重新编写全部代码。只要修改一小部分就可以完美实现Ajax带来的无闪刷新快感。 

以上代码均在IE,FireFox下测试过! 

首先建立一个数据表menu 

mId  菜单主键 
name 菜单名称 
url  菜单链接 
father 低级菜单ID 
sub  是否最底层菜单(用于判断是否还可以继续展开) 
target 菜单链接目标(用ajax方式打开时作为显示id) 
pa   菜单参数(这项用于ajax方式打开菜单) 

制作一个菜单对象类 


class Menu{ 
private int mId; 
private String name; 
...//其它成员 

public getMid(){ 
return mId; 

public setMid(int mId){ 
this.mId = mId; 

....//其它成员的get set方法, 



另一个是操作类 

class MenuOpt(){ 
public Vector getMenus(int father){ 
Vector vector = new Vector(); 
//这里是取得父级菜单ID为father的全部菜单 
//并封装进Vector的一个对象中。。 
return vector; 


其次就是一般的jsp文件了。但要注意以前说过的,不要包含标签! 
menu.jsp: 

<%@page contentType="text/html; charset=GB2312"%> 
<%@taglib uri="" prefix="c"%> 
 
 
 

 
 
 
 
 
 
${m.name} 
 
 
 
 
 
 
 
 
${m.name} 
 
 
 
 
 
menu.js: 

//operMenu(打开下拉菜单的ID,打开的地址,链接打开的目标,参数)。 
//这是用在menu.jsp的方法 
function showMenu(id,url,target,param){ 
var trObj = document.getElementById("tr"+id); 
var tdObj = document.getElementById(id); 
//try{ 
if(document.getElementById("tr"+id).style.display == "none"){ 
//显示菜单 
if(tdObj.innerHTML == null || tdObj.innerHTML == ""){ 
//提取数据 
document.getElementById("tr"+id).style.display = ""; 
document.getElementById("img"+id).src = "pic/menu2.gif" 
Bcandy(id,"page/menu.jsp",param,""); 
openMenu(url,target,param); 
}else{ 
//如果里面有内容,直接显示 
document.getElementById("tr"+id).style.display = ""; 
document.getElementById("img"+id).src = "pic/menu2.gif" 
openMenu(url,target,param); 

//Bcandy(target,url,param,"");//打开菜单链接 
}else{ 
//隐藏菜单 
document.getElementById("tr"+id).style.display = "none"; 
document.getElementById("img"+id).src = "pic/menu0.gif" 

//}catch(e){} 


//打开菜单 
function openMenu(url,target,param){ 
//这里不用我写了吧。有好几种实现方法,建议使用ajax实现! 

最后是显示页面


<%@ page contentType="text/html; charset=GB2312" %>



function ini(){
Bcandy("0","menu.jsp","id=0&father=0","menu.js");
}







可以看到,无论在哪个层面,都和传统的没什么分别,只有jsp部分除去文件头而已(其实不去掉也行的,呵呵),而且,还可以看到,一个页面,已经分成了好几部分。就像之前说的那样,积木式的(这是网上看到一篇关于.net框架的结构时作者提出的一种结构,觉得不错,被我应用到JSP来了)。

在一些细节方面,我作了一些保留,请理解。但大致框架都是经过IE和FireFox测试。一些功能方面的扩展,自己想想了。

原理:其实就是应用了页面递归!就和一般的递归方法一下,不过用在页面上而已


循环,将从封装进vector的对象逐一显示出来
for{
 if(如果是最上层菜单sub=N){
 

  显示菜单内容
 

 
 
 }else{
  
显示菜单内容

 }
}

showMenu(father,id....)方法,将根据传入的father去服务器里取得数据后,再次调用这个页面。而这时,是将页面的内容显示在新的ID里面。这样,看起来就有和MSDN里的树菜单一样的效果了。

优点:多级菜单多次获取,加快了反应速度,同时应用了ajax请求,让人感觉不到页面的闪烁,亲和力强。再者,可以JS里加入了代码,让用户不用每次点击都去获取服务器数据,而是先判断有没有内容,没有再取。。。同时,实现了菜单与页面的同步,在每打开一级菜单,都可以在相应的地方打开页面。同样,这个operMenu()也可以采用ajax方式。

效果可以上 看看

阅读(931) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~