<p> 不知上过ChinaRen校友录的朋友们有没有注意,ChinaRen在改版后很多方面都进行了较大的改动。例如留言与回复方面已经不再像以前那样,在每次提交后总得重新加载一下整个页面,碰到网速超慢时,只能干瞪着眼睛等待。那么现在这种炫丽的效果又是如何做到的呢,如果你觉得有兴趣,那就跟着我一起往下看吧!</p><p> 了解Ajax的朋友也许知道其实这种炫丽效果的实现并不是一件很为难的事,当然如果你不了解何为Ajax那也没有关系,本节我们就通过打造一个新闻评论系统,来看看到底什么是Ajax,首先我们还是先了解一些基础东西。</p><p> 什么是Ajax?</p><p> Ajax提出者Jesse James Garrett在《Ajax:一个Web应用的新途径》中提到Ajax为“Asynchronous JavaScript + XML”的简称,也就是异步的JavaScript和XML处理。其包含:</p><p> 用XHTML和CSS进行基于标准的表示方式:</p><p> 采用DOM(document object model)动态显示和交互操作;</p><p> 采用XML和XSLT进行数据交换和操作;</p><p> 采用XMLHttpRequest进行异步数据获取;</p><p> 采用JavaScript绑定上述技术应用;</p><p> Ajax与传统Web应用有什么不同?</p><p> Ajax与传统Web应用最大的不同就是Ajax可以对页面某一个区域进行局部加载,而不是像传统Web中每次页面请求后的都必须重新加载整个页面,特别在页面负载比较大的情况下,页面加载时间就比较长,用户多数时间就处在等待状态,而呈现给用户的仅仅是一片空白,而在Ajax的应用中就可以很好的避免这类事情的发生。</p><p> Ajax的工作原理是什么?</p><p> Ajax主要是通过JavaScript对象中的XmlHttpRequest向服务器提出请求,并根据处理的结果更新页面。这样的更新不会使整个页面全部更新,而是根据用户的需要对某个区域进行局部更新,而且在更新的同时不影响其它区域的浏览。例如:搜狐个人博客中每个栏目后面的刷新按钮。</p>
<p> </p>
<p> 什么是XmlDom?</p><p> XMLDOM是用来访问和操作XML文档的编程接口规范。XMLDOM被设计为可用于任何语言和任何操作系统。借助DOM,程序员可以创建XML文档、遍历其结构,增、改、删其元素。DOM将整个XML文档视作一棵树,文档级的元素是树的根。</p><p> 下面我们就来看看几个与本教程相关的功能,值得注意的是以下方法或属性并不是同一个对象下,具体请看说明:</p><p> getElementsByTagName 方法</p><p> 说明:传回指定名称的元素集合。</p><p> 语法:objNodeList = xmlDocument.getElementsByTagName(tagname);</p><p> 例:var node=xmlDom.responseXML.getElementsByTagName("pl");</p><p> getAttribute()方法</p><p> 说明:获得某个元素节点的属性值</p><p> 语法:elementNode.getAttribute(name)</p><p> 例:var tot=xmlDom.responseXML.getElementsByTagName("pl")[0].getAttribute("tot");</p><p> childNodes 属性</p><p> 说明:传回一个节点列表,包含该节点所有可用的子节点。</p><p> 语法:objNodeList=node.childNodes;</p><p> 例:objNodeList = xmlDoc.childNodes;</p><p> 如需具体的某一个节点, var u= xmlDoc.childNodes(0);</p><p> Length属性 ()</p><p> 说明:返回一个节点列表中的节点数量</p><p> 语法:nodelistObject.length</p><p> 例:var len=node.length;</p><p> 至此,基础知识已经讲完了,如果你还是不太了解的话,那建议你去看一些javascript相关的教程。下面我们就看这个新闻评论系统的具体实现原理</p><p> 假设有一个页面index.asp,上半部分为评论列表显示区域,下面为评论提交区域。那么这样一个页面我们如何显示评论内容和提交评论呢?</p><p> 传统:上半部分评论列表直接通过数据库查询语句读取并显示,每当提交新的评论时,先传递给处理页面,处理页面处理完毕后再返回index.asp这个页面,当然index.asp是重新加载获得新的评论。</p>
<p> </p>
<p> Ajax::首先列表页面的内容是一个单独的xml文件(pl_list.asp),然后index..asp中的上半部分评论通过XmlHttpRequest请求pl_list.asp页面,并通过返回的结果传递到需要更新区域。提交评论同样如此,每次提交采用XmlHttpRequest请求提交处理程序,然后重新更新评论列表显示区域。</p><p> 此新闻评论系统共分为五个部分,分别为数据库、前台页面、JS代码、服务器处理、CSS样式。</p><p> 数据库的设计</p><p> PL表:</p><p> 字段名</p><p> 类型</p><p> 长度</p><p> id</p><p> 自动编号</p><p> user</p><p> 文本</p><p> 20</p><p> dateandtime</p><p> 日期/时间</p><p> content</p><p> 备注</p><p> newid</p><p> 数字</p><p> 前台页面:(index.htm)</p><p> 如上图所示,前台页面共包括两部分,上半部分为页面评论列表显示,下半部分为提交评论。由于我们这里只是模拟一个新闻评论系统,并没有真正的新闻页面,那么在传递新闻ID的时候我们采用了一个默认值 〈input name="newsid" value="1" type="hidden"/〉。</p><p> 代码:index.htm</p><code> 〈%@LANGUAGE="VBSCRIPT" CODEPAGE="936"%〉<br /> 〈!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"〉<br /> 〈html xmlns="http://www.w3.org/1999/xhtml"〉<br /> 〈head〉<br /> 〈meta http-equiv="Content-Type" content="text/html; charset=gb2312" /〉<br /> 〈title〉评论系统〈/title〉<br /> 〈script src="main.js"〉〈/script〉<br /> 〈link href="main.css" rel="stylesheet" type="text/css" /〉<br /> 〈/head〉<br /> 〈body〉<br /> 〈div id="pllist"〉正在加载评论……<br /> 〈script〉 loadDom();setTimeout("loadDom()",10000);〈/script〉<br /> 〈/div〉<br /> 〈div style="width:240px;font-size:12px;text-align:center"〉<br /> 〈fieldset〉〈legend〉评论〈/legend〉<br /> 呢称:〈input name="user" type="text" style="width:180px"/〉〈input name="newsid" value="1" type="hidden"/〉〈br/〉<br /> 内容:〈textarea name="content" style="width:180px;height:80px"〉〈/textarea〉〈br/〉<br /> 〈input name="submit" value="我要评论" onclick="fb();" type="button" /〉<br /> 〈/fieldset〉<br /> 〈/div〉<br /> 〈div style="font-size:12px;" id="msg"〉<br /> 〈/div〉<br /> 〈/body〉<br /> 〈/html〉</code><p> JS代码页(核心部分) main.js</p>
<p> </p>
<p> JS代码算是本系统的一个核心部分了,Ajax的体现基本全包含在这短短数十行的代码中,是连结前台与后台处理的一个桥梁,可谓是重中之重,为了更好的让大家理解整个功能,我们将分段介绍。</p><p> 1、获得XmlHttp对象,创建并返回一个XmlHttp对象。</p><code> var xhr;<br /> function getXHR()<br /> {<br /> try {<br /> xhr=new ActiveXObject("Msxml2.XMLHTTP");<br /> } catch (e) {<br /> try {<br /> xhr=new ActiveXObject("Microsoft.XMLHTTP");<br /> } catch (e) {<br /> xhr=false;<br /> }<br /> }<br /> if(!xhr&&typeof XMLHttpRequest!='undefined')<br /> {<br /> xhr=new XMLHttpRequest();<br /> }<br /> return xhr;<br /> }<br /> function openXHR(method,url,callback)<br /> {<br /> getXHR();<br /> xhr.open(method,url);<br /> xhr.onreadystatechange=function()<br /> {<br /> if(xhr.readyState!=4)return;<br /> callback(xhr);<br /> }<br /> xhr.send(null);<br /> }<br /> function loadXML(method,url,callback)<br /> {<br /> getXHR();<br /> xhr.open(method,url);<br /> xhr.setRequestHeader("Content-Type","text/xml");<br /> xhr.setRequestHeader("Content-Type","GBK");<br /> xhr.onreadystatechange=function()<br /> {<br /> if(xhr.readyState!=4)return;<br /> callback(xhr);<br /> }<br /> xhr.send(null);<br /> }</code><p> 具体的调用方法:</p>
<p> </p>
<p> loadXML(method,url,callback)</p><p> method: http方法,例如:POST、GET、PUT及PROPFIND</p><p> url: 请求的URL地址,可以为绝对地址也可以为相对地址</p><p> callback:自定义的返回处理函数</p><p> 2.获得评论列表</p><p> 此段代码的主要功能是根据服务器处理返回的信息更新前台页面的内容,主要包括显示评论列表、评论列表分页、跳转页数三个功能。</p><p> 显示评论列表:getList函数</p><code> function getList(xmlDom)<br /> {<br /> var pllist=document.getElementById("pllist"); //获得页面pllist对象,此对象用来显示评论内容<br /> var node=xmlDom.responseXML.getElementsByTagName("pllist");//获得pllist节点集合<br /> var tot=xmlDom.responseXML.getElementsByTagName("pl")[0].getAttribute("tot");//获得pl节点tot属性值,这里指评论的总数量<br /> var curpage=xmlDom.responseXML.getElementsByTagName("pl")[0].getAttribute("curpage");//获得pl节点curpage属性,这里指评论列表当前所在的页数,应用于翻页<br /> if (tot!=0) //判断当前评论数是否为空<br /> {<br /> var cont="";<br /> var len=node.length;//获得pllist节点集合中节点的总数量<br /> for(var i=0;i〈len;i++)<br /> {<br /> var u=node[i].childNodes(0).text;//获得节点第一个子节点的值,这里指呢称<br /> var d=node[i].childNodes(1).text; //获得节点第二个子节点的值,这里指时间<br /> var co=node[i].childNodes(2).text; //获得节点第三个子节点的值,这里指内容<br /> var idnub=node[i].childNodes(3).text; //获得节点第四个子节点的值,这里指新闻ID<br /> cont+='〈div class="u"〉呢称:'+u+'〈/div〉〈div class="d"〉时间:'+d+'〈/div〉〈div class="idnub" onclick="del('+idnub+')" style="cursor:hand" onmouseout="this.style.background=''" onmousemove="this.style.background='#99cc66'"〉删除〈/div〉〈div class="co"〉内容:'+co+'〈/div〉'; //将所获得的评论内容生成一个字符串<br /> }<br /> var cont1=pagecount(tot,curpage);//调用分页函数<br /> cont+=cont1;<br /> pllist.innerHTML=cont;//将内容呈现<br /> }<br /> else<br /> {<br /> pllist.innerHTML="暂无评论!";<br /> }<br /> }</code><p> 评论列表分页:pagecount函数</p>
<p> </p>
<code> function pagecount(tot,cur)<br /> {<br /> var cont1="";<br /> if (tot%5==0) //默认每页五条,这个要求与服务器端保持一致<br /> {<br /> pages=parseInt(tot/5);<br /> }<br /> else<br /> {<br /> pages=parseInt(tot/5)+1;<br /> }<br /> for(var j=1;j〈=pages;j++)<br /> {<br /> if (j==cur)<br /> { cont1+="〈span〉"+j+"〈/span〉 "}<br /> else<br /> { cont1+="〈span style='cursor:hand;color:#0000ff' onmouseout='this.style.background=""' onmousemove='this.style.background="#99cc66"' onclick='gotopage("+j+")'〉"+j+"〈/span〉 "}<br /> }<br /> return cont1;<br /> }</code><p> 跳转页数:gotopage函数</p><code> function gotopage(page)<br /> {<br /> loadXML("get","pl_list.asp?page="+page,getList);<br /> }<br /> function loadDom() //定时更新评论列表,初始化10秒钟<br /> {<br /> loadXML("get","pl_list.asp",getList);<br /> setTimeout("loadDom()",10000)<br /> }</code><p> 3.删除评论</p><code> function del(idnub)<br /> {<br /> var msg=document.getElementById("msg");<br /> msg.innerText="正在删除……";<br /> loadXML("get","pl_del.asp?id="+idnub,getdel);<br /> }<br /> function getdel(xmlDom) //删除后所触发的事件,更新页面<br /> {<br /> var msg=document.getElementById("msg");<br /> msg.innerText="删除成功!";<br /> loadXML("get","pl_list.asp",getList);<br /> }</code><p> 4.提交评论</p>
<p> </p>
<code> function fb() //处理提交<br /> {<br /> var msg=document.getElementById("msg");<br /> var user=document.getElementById("user");<br /> var content=document.getElementById("content")<br /> var newsid=document.getElementById("newsid")<br /> if (user.value=="")<br /> {<br /> alert("呢称不可为空!");<br /> return false;<br /> }<br /> if (content.value=="")<br /> {<br /> alert("内容不可为空!");<br /> return false;<br /> }<br /> msg.innerText="正在发表评论";<br /> loadXML("get","pl_fb.asp?user="+user.value+"&content="+content.value+"&newsid="+newsid.value,getfb);<br /> }<br /> function getfb(xmlDom) //评论提交后所触发的事件,更新评论列表<br /> {<br /> var msg=document.getElementById("msg");<br /> msg.innerText=xmlDom.responseText;<br /> loadXML("get","pl_list.asp",getList);<br /> }</code><p> 服务器处理程序</p><p> 根据JS代码页的分段介绍,我们了解此系统的功能大致包括评论的显示处理、评论的删除处理、评论的提交处理三个功能,那么我们就根据这三个功能分别介绍。</p><p> 评论的显示处理页面:pl_list.asp</p><p> 此程序为asp生成xml文件,通过分页的方式将评论的内容以XML的形式呈现出来,我们可以单独运行。</p><p> 代码:</p><code> 〈!--#include file="conn.asp"--〉<br /> 〈%<br /> Response.ContentType = "text/XML"<br /> Response.expires = 0<br /> Response.expiresabsolute = Now() - 1<br /> Response.addHeader "pragma", "no-cache"<br /> Response.addHeader "cache-control", "private"<br /> Response.CacheControl = "no-cache"<br /> Response.write("〈?xml version=""1.0"" encoding=""gb2312""?〉")<br /> currentpage=request("page")<br /> if currentpage="" or int(currentpage)=0 then currentpage=1<br /> set rs=server.createobject("adodb.recordset")<br /> sql="select * from pl order by id desc"<br /> rs.cursorlocation=3<br /> rs.open sql,conn,1,1<br /> if not rs.bof or not rs.eof then<br /> rs.pagesize=5<br /> rs.absolutepage=currentpage<br /> rowcount=rs.pagesize<br /> Response.write("〈pl tot='"&rs.recordcount&"' curpage='"¤tpage&"'〉")<br /> do while not rs.eof and rowcount〉0<br /> Response.write("〈pllist〉")<br /> Response.write("〈user〉"&rs("user")&"〈/user〉")<br /> Response.write("〈dateandtime〉"&rs("dateandtime")&"〈/dateandtime〉")<br /> Response.write("〈content〉"&rs("content")&"〈/content〉")<br /> Response.write("〈id〉"&rs("id")&"〈/id〉")<br /> Response.write("〈/pllist〉")<br /> rowcount=rowcount-1<br /> rs.movenext<br /> loop<br /> else<br /> Response.write("〈pl tot='"&rs.recordcount&"' curpage='"¤tpage&"'〉")<br /> end if<br /> rs.close<br /> set rs=nothing<br /> response.write("〈/pl〉")<br /> %〉</code><p> Conn.asp 数据库连结文件,在删除与提交处理中同样使用</p>
<p> </p>
<code> 〈%<br /> dim conn<br /> dim connstr<br /> dim db<br /> db="main.mdb" '数据库文件位置<br /> connstr="Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" & Server.MapPath(""&db&"")<br /> set conn=server.createobject("ADODB.CONNECTION")<br /> conn.open connstr<br /> %〉</code><p> 评论的删除处理页面:pl_list.asp</p><code> 〈% Response.Charset="gb2312" %〉<br /> 〈% Session.CodePage=936 %〉<br /> 〈!--#include file="conn.asp"--〉<br /> 〈%<br /> id=request("id")<br /> if id="" then<br /> response.write("参数错误!")<br /> response.End()<br /> end if<br /> set rs=server.CreateObject("adodb.recordset")<br /> sql="select * from pl where id="&id<br /> rs.open sql,conn,1,3<br /> rs.delete<br /> rs.update<br /> rs.close<br /> set rs=nothing<br /> response.write("删除成功!")<br /> %〉</code><p> 评论的提交处理页面:pl_fb.asp</p><code> 〈% Response.Charset="gb2312" %〉<br /> 〈% Session.CodePage=936 %〉<br /> 〈!--#include file="conn.asp"--〉<br /> 〈%<br /> user=request("user")<br /> content=request("content")<br /> newsid=request("newsid")<br /> set rs=server.CreateObject("adodb.recordset")<br /> sql="select * from pl"<br /> rs.open sql,conn,1,3<br /> rs.addnew<br /> rs("user")=user<br /> rs("content")=content<br /> rs("newsid")=newsid<br /> rs("dateandtime")=time()<br /> rs.update<br /> rs.close<br /> set rs=nothing<br /> response.write("添加成功!")<br /> %〉</code><p> CSS样式 main.css</p></p><p> 一个好的页面呈现效果离不开一个好的样式,当然我这个属于最基本的,算是看得清楚罢了,大家如果有兴趣可以对样式文件作修改。</p><code> .u { /*呢称*/<br /> font-size: 12px;<br /> float:left;<br /> height:25px;<br /> line-height:20px;<br /> width:120px;<br /> }<br /> .d { /*时间*/<br /> font-size: 12px;<br /> float:left;<br /> height:25px;<br /> line-height:20px;<br /> width:120px;<br /> }<br /> .idnub { /*删除*/<br /> text-align:center;<br /> font-size: 12px;<br /> height:25px;<br /> line-height:25px;<br /> width:30px;<br /> }<br /> .co {/*内容*/<br /> font-size: 12px;<br /> width:280px;<br /> }</code><p> 这是我第一次尝试在asp中使用ajax,习惯于donet中的拖拖拉拉,突然用最原始的方式写代码还真有点不习惯,特别是缺少了那种所见即所得的效果,每一步的实现都得在不断的调试中完成,的确是一件很累的事情。当然随着Ajax式的程序开发愈来愈受欢迎,现在市面的框架、工具也越来越多,相信有那么一天,开发Ajax应用程序终究会变成一件易常简单的事。</p></p>