响应速度快。
复杂的UI成功处理,一直以来,我们对B/S模式的UI不如C/S模式UI丰富而苦恼,现在由于AJAX大量使用JS,使得复杂的UI设计变得更加成功。
最后AJAX请求的返回对象为XML文件,易于和WEB SERVICE结合起来。
基于Servlet为后台的一个web应用例子:
关于关联选择框的问题:
当用户在第一个选择框里选择ZHEJIANGS时,第二个选择框要出现ZHEJIANG的城市;当用户在第一个选择框里选择JIANGSU时,第二个选择框里要出现JIANGSU的城市
首先,配置文件web.xml,在里面配置servlet,跟往常一样:
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"> <servlet> <servlet-name>SelectCityServlet</servlet-name> <servlet-class>com.stephen.servlet.SelectCityServlet</servlet-class> </servlet> <servlet-mapping> <servlet-name>SelectCityServlet</servlet-name> <url-pattern>/servlet/SelectCityServlet</url-pattern> </servlet-mapping> </web-app> JSP文件: <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <title>MyHtml.html</title> <meta http-equiv="keywords" content="keyword1,keyword2,keyword3"> <meta http-equiv="description" content="this is my page"> <!--link rel="stylesheet" type="text/css" href="./style.css">HTML显式注释--> </head> <script type="text/javascript"> function getResult(stateVal){ var url = "servlet/ServletCityServlet?state="+stateVal; if(window.XMLHttpRequest){//code for all new browsers req = new XMLHttpRequest(); }else if(window.ActiveXObject){//code for IE5 IE6 req = new ActiveXObject("Microsoft.XMLHTTP"); } if(req){ req.open("GET",url,true); req.onreadystatechange = complete; req.send(null); } } function complete(){ if(req.readyState == 4){//4 = "loaded" if(req.status == 200){//200 = OK var city = req.responseXML.getElementsByTagName("city"); //alert(city.length) var str = new Array(); for(var i=0;i<city.length;i++){ str[i]=city[i].firstChild.data; } //alert(document.getElementById("city")); buildSelect(str,document.getElementById("city")); } } } function buildSelect(str,sel){ sel.options.length=0; for(var i=0;i<str.length;i++){ sel.options[sel.options.length] = new Option(str[i],str[i]) } } </script> <body> <select name="state" onChange="getResult(this.value)"> <option value="">Select</option> <option value="zj">ZEHJIANG</option> <option value="js">JIANGSU</option> </select> <select id="city"> <option value="">CITY</option> </select> </body> </html>
JS的不同
getResult(stateVal)方法,取得XmlHttpRequest;然后设置该请求的url:req.open("GET",url,true);接着设置请求返回值的接收方法:req.onreadystatechange = complete;该返回值的接收方法为 complete();最后是发送请求:req.send(null);
然后是返回值接收方法:complete(),这个方法,首先判断是否正确返回,如正确返回,用DOM对返回的XML文件进行解析。得到city值后,再通过buildSelect(str,sel)方法赋值到相应的选择框里面。
onreadystatechange是一个事件句柄,他的值complete是一个函数名称,当XMLHttpRequest对象的状态发生改变时,会触发此函数。状态从0到4进行变化。仅在状态为4时,才执行代码。
为什么使用Async=true?
open()的第三个参数中使用了"true"
该参数规定请求是否异步处理。
True表示脚本会在send()方法之后继续运行,而不等待来自服务器的响应。
onreadystatechange事件使代码复杂化了。但是这是在没有得到服务器响应的情况下,防止代码停止的最安全的方法。
通过把该参数设置为"false",可以省去额外的onreadystatechange代码。如果在请求失败时是否执行其余的代码无关紧要,那么就可以使用这个参数。
Servlet:
import java.io.IOException; import java.io.PrintWriter; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class SelectCityServlet extends HttpServlrt{ public SelectCityServlet(){ super(); } public void destory(){ super.destory(); } public void doGet(HttpServletRequest request,HttpServletResponse response)throws ServletException IOException{ response.setContentType("text/xml"); response.setHeader("Cache-Control","no-cache"); String state = request.getParamter("state"); StringBuffer sb = new StringBuffer("<state>"); if("zj".equals(state)){ sb.append("<city>hangzhou</city><city>huzhou</city>"); }else if("js".equals(state)){ sb.append("<city>nanjing</city><city>wuxi</city>"); } sb.append("</state>"); PrintWriter out = response.getWriter(); out.write(sb.toString()); //out.write只能输出字符串,out.print可以输出Java对象 out.close; } }
这个类,首先是从request里取得state参数,然后根据state参数生成相应的XML文件,最后将XML文件输出到PrintWriter对象里。