Ajax核心技术之XMLHttpRequest



      Ajax:即“Asynchronous JavaScript and XML”(异步JavaScript和XML),一门综合性的技术:运用JavaScript对象XMLHttpRequest进行异步数据交换;JavaScript操作DOM实现动态效果;运用XHTML+CSS表达信息;XMLXSLT操作数据。此篇文章重点介绍使用XMLHttpRequest对象与服务器端进行异步数据交换。

使用方法

        XMLHttpRequest五步使用法:

               1.创建对象;

               2.注册回调函数;

               3.使用open方法设置和服务器交互的基本信息;

               4.设置发送的数据,开始和服务器端交互;

       5.实现回调函数.

        由于每次应用XMLHttpRequest对象时,都要进行五步操作,因此,可将该对象的使用封装为js文件中,传递部分参数使用其方法就可以完成相应功能,实现如下:

//使用封装方法人员只提供http的请求,url地址,数据,成功和失败的回调的方法

//1.定义XMLHttpRequest对象的构造方法
var MyXMLHttpRequest =function(){
    var xmlhttprequest;
    if(window.XMLHttpRequest){
        //IE7,IE8,FireFox,Mozillar,Safari,Opera
        //alert("IE7,IE8,FireFox,Mozillar,Safari,Opera");
        xmlhttprequest = new XMLHttpRequest();
        //解决浏览器在服务器端响应由于没有Text头的时候可能无法工作的问题
        if(xmlhttprequest.overrideMimeType){
            xmlhttprequest.overrideMimeType("text/xml");
        }
    }else if(window.ActiveXObject){
        //IE6,IE5.5,IE5
        alert("IE6,IE5.5,IE5");
        var activexName =["MSXML2.XMLHTTP","Microsoft.XMLHTTP"];
        for (var n=0;n<activexName.length;n++){
            try{
                xmlhttprequest = new ActiveXObject(activexName[i]);
                break;
            }catch(e){
                
            }
        }
    }
    if(xmlhttprequest === null){
        alert("当前浏览器不支持创建XMLHttpRequest对象,请更换浏览器");
    } else {
        this.xmlhttp=xmlhttprequest;
    }
};
//用户发送请求的方法
MyXMLHttpRequest.prototype.send = function(method,url,data,callback,failback){
    if (this.xmlhttp !== undefined && this.xmlhttp !== null){
        method = method.toUpperCase();
        if(method !== "POST" && method !== "GET") {
            alert("HTTP的请求方式必须是GET或POST");
            return;
        } 
        if(url === null || url === undefined) {
            alert("HTTP的请求地址必须设置");
            return;
        }
        var tmpxmlhttp = this.xmlhttp;
        //2.注册回调函数
        this.xmlhttp.onreadystatechange = function(){
            //5.判断和服务器的交互是否完成,还要判断服务器端是否正确返回了数据
            if (tmpxmlhttp.readyState == 4) {
                //表示和服务器的交互已经完成
                if(tmpxmlhttp.status == 200){
                    //表示服务器的相应代码是200,正确的返回了数据
                    //纯文本数据的接收方法
                    var responseText = tmpxmlhttp.responseText;
                    var responseXML = tmpxmlhttp.responseXML;
                    if(callback === undefined || callback === null) {
                        alert("没有设置处理数据正确返回的方法");
                        alert("返回的数据:" + responseText);
                    } else {
                        callback(responseText,responseXML);
                    }
                }else {
                    if(failback === undefined || failback === null){
                        alert("没有设置处理数据返回的处理方法");
                        alert("Http的相应码:" + tmpxmlhttp.status + "相应码的文本信息:" + xml.statusText);
                    } else {
                        failback(tempxmlhttp.status,tempxmlhttp.statusText);
                    }
                }
            }
        }
        //3.使用open方法设置和服务器端交互的基本信息
        this.xmlhttp.open(method,url,true);
        //如果是POST方式,需要设置请求头
        if(method === "POST") {
            this.xmlhttp.setRequestHeader("Content-Type","application/x-www-form-urlencode");
        }
        //4.设置发送的数据,开始和服务器端交互。
        this.xmlhttp.send(data);
    } else{
        alert("XMLHttpRequest对象创建失败,无法数据");
    }
}
//取消当前响应,关闭连接并且结束任何未决的网络活动
MyXMLHttpRequest.prototype.abort = function(){
    this.xmlhttp.abort();
}


扩展问题

         1.浏览器缓存

         2.中文乱码

         3.跨域访问

对于问题1、问题3都可以通过更改url地址的方法得以解决。问题1可在url地址尾添加时间戳,问题3通过代理方式进行解决。只需在send()中的第三步执行前添加相应判断即可:

        //解决缓存的转换:增加时间戳
        if(url.indexOf("?") >= 0 ){
            url = url + "&t=" + (new Date()).valueOf();
        } else {
            url = url + "?t=" + (new Date()).valueOf();
        }
        
        //解决跨域的问题
        if(url.indexOf("http://") >= 0) {
            url.replace("?","&");
            url = "Proxy?url=" + url;
        }

问题3对应代理服务端实现:


    /**
     * Handles the HTTP <code>GET</code> method.
     *
     * @param request servlet request
     * @param response servlet response
     * @throws ServletException if a servlet-specific error occurs
     * @throws IOException if an I/O error occurs
     */
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        //获取参数,最后得到请求url地址类似于:url = http://192.168.../AJAX/AJAXServer?aa=11&bb=22&cc=33
        StringBuilder url = new StringBuilder();
        url.append(request.getParameter("url"));
        //获取访问的跨域地址url = http://192.168.../AJAX/AJAXServer
        Enumeration enu = request.getParameterNames();
        boolean flag = false;       //定义标志变量,表示是否为拼接的第一个参数
        while(enu.hasMoreElements()){
                String paramName = (String) enu.nextElement();
                if(!paramName.equals("url")){
                        String paramValue = request.getParameter(paramName);
                        paramValue = URLEncoder.encode(paramValue,"utf-8");
                        if(!flag){
                                url.append("?").append(paramName).append("=").append(paramValue);
                                flag = true;
                        } else {
                                url.append("&").append(paramName).append("=").append(paramValue);
                        }
                }
        }
        response.setContentType("text/html;charset=utf-8");
        PrintWriter out = response.getWriter();
        if(url != null && url.length() > 0){
                URL connectionUrl = new URL(url.toString());
                BufferedReader reader = new BufferedReader(new InputStreamReader(connectionUrl.openStream(),"utf-8"));
                String line;
                while((line = reader.readLine()) != null){
                        out.println(line);
                }
                reader.close();
        }
    }

    /**
     * Handles the HTTP <code>POST</code> method.
     *
     * @param request servlet request
     * @param response servlet response
     * @throws ServletException if a servlet-specific error occurs
     * @throws IOException if an I/O error occurs
     */
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        //获取参数,最后得到请求url地址类似于:url = http://192.168.../AJAX/AJAXServer?aa=11&bb=22&cc=33
        String url = request.getParameter("url");
        //获取访问的跨域地址url = http://192.168.../AJAX/AJAXServer
        StringBuffer paramBuffer = new StringBuffer();
        Enumeration enu = request.getParameterNames();
        boolean flag = false;       //定义标志变量,表示是否为拼接的第一个参数
        while(enu.hasMoreElements()){
                String paramName = (String) enu.nextElement();
                if(!paramName.equals("url")){
                        String paramValue = request.getParameter(paramName);
                        paramValue = URLEncoder.encode(paramValue,"utf-8");
                        if(!flag){
                                paramBuffer.append(paramName).append("=").append(paramValue);
                                flag = true;
                        } else {
                                paramBuffer.append("&").append(paramName).append("=").append(paramValue);
                        }
                }
        }

        response.setContentType("text/html;charset=utf-8");
        PrintWriter out = response.getWriter();
        if(url != null && url.trim().length() > 0){
                URL connectUrl = new URL(url);
                URLConnection connection =connectUrl.openConnection();
                connection.setDoOutput(true);
                OutputStreamWriter writer = new OutputStreamWriter(connection.getOutputStream());
                writer.write(paramBuffer.toString());
                writer.flush();
                BufferedReader reader = new BufferedReader(new InputStreamReader(connection.getInputStream(),"utf-8"));
                String line;
                while((line = reader.readLine()) != null){
                        out.println(line);
                }
                writer.close();
                reader.close();
        }
    }


        对于问题2


        (1)响应数据解决中文乱码的方法:保证页面端定义的charsethttp响应头的Content-Type中定义的charset一致。


        http响应头的Content-Typecharset设置为utf-8


        ②仅仅使用"MSXML2.XMLHTTP","Miscrosoft.XMLHTTP"这两个中的一个来创建XMLHttpRequest对象。


        (2)请求数据中文乱码解决方法:由于服务器端getParameter()方法会自动进行一次ISO8859规则解码,因此页面端应使用javascript中的encodeURI()方法对请求数据进行两次编码,服务器端使用URLDecode(utf-8)进行一次解码。


总结

         XMLHttp详细属性、方法介绍:http://www.w3school.com.cn/xmldom/dom_http.asp

   Ajax异步交互主要通过XMLHttpRequest对象实现,搜索SuggestGoogleMap等应用的友好体验使得此对象的使用愈显重要,希望本文对读者更好的使用XMLHttpRequest提供了一定帮助。

   



评论 23
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值