Ajax

一 概述

对于web应用程序,用户浏览器发送请求,服务器接收并处理请求,然后返回结果,往往返回就是字符串(HTML),浏览器将字符串(HTML)渲染并显示浏览器上。

客户端是怎么向服务端发送数据的呢?

1.1 传统方式

  • GET:地址栏、a标签、Form表单
  • POST:Form表单

通过以上方式,一个简单操作需要重新加载全局数据。

1.2 Ajax

AJAX,Asynchronous JavaScript and XML (异步的JavaScript和XML),一种创建交互式网页应用的网页开发技术方案。

  • 异步的JavaScript
    使用JavaScript语言以及相关浏览器提供类库的功能向服务端发送请求,当服务端处理完请求之后,自动执行某个JavaScript的回调函数
    PS:以上请求和响应的整个过程是偷偷进行的,页面上无任何感知。
  • XML
    XML是一种标记语言,是Ajax在和后台交互时传输数据的格式之一

同步交互与异步交互

  • 同步交互:客户端发出一个请求后,需要等待服务器响应结束后,才能发出第二个请求;
  • 异步交互:客户端发出一个请求后,无需等待服务器响应结束,就可以发出第二个请求。

Ajax的特点:

  • 异步交互:当请求发出后,浏览器还可以进行其他操作,无需等待服务器的响应
  • 局部刷新: 整个过程中页面没有刷新,只是刷新页面中的局部位置而已(服务器相应内容为页面中的局部),性能高

Ajax的应用场景:

  • 注册时,输入用户名自动检测用户是否已经存在
  • 登陆时,提示用户名密码错误
  • 删除数据行时,将行ID发送到后台,后台在数据库中删除,数据库删除成功后,在页面DOM中将数据行也删除

举一个简单的例子:

当我们在百度搜索框中输入一个“数据”后,会马上出现一个下拉列表!列表中显示的是包含该数据的4个关键词。

其实这就使用了AJAX技术!当搜索框发生了输入变化时,浏览器会使用AJAX技术向服务器发送一个请求,查询包含该数据的前4个关键词,然后服务器会把查询到的结果响应给浏览器,最后浏览器把这4个关键字显示在下拉列表中。

  • 整个过程中页面没有刷新,只是刷新页面中的局部位置而已!
  • 当请求发出后,浏览器还可以进行其他操作,无需等待服务器的响应!

二 原生Ajax

Ajax主要就是使用 XMLHttpRequest对象来完成请求的操作,该对象在主流浏览器中均存在(除早期的IE),Ajax首次出现IE5.5中存在(ActiveX控件)。

2.1 XMLHttpRequest对象介绍

XMLHttpRequest对象的主要方法:

a. void open(String method,String url,Boolen async)
   用于创建请求
    
   参数:
       method: 请求方式(字符串类型),如:POST、GET、DELETE...
       url:    要请求的地址(字符串类型)
       async:  是否异步(布尔类型)
 
b. void send(String body)
    用于发送请求
 
    参数:
        body: 要发送的数据(字符串类型)
 
c. void setRequestHeader(String header,String value)
    用于设置请求头
 
    参数:
        header: 请求头的key(字符串类型)
        vlaue:  请求头的value(字符串类型)
 
d. String getAllResponseHeaders()
    获取所有响应头
 
    返回值:
        响应头数据(字符串类型)
 
e. String getResponseHeader(String header)
    获取响应头中指定header的值
 
    参数:
        header: 响应头的key(字符串类型)
 
    返回值:
        响应头中指定的header对应的值
 
f. void abort()
    终止请求

XMLHttpRequest对象的主要属性:

a. Number readyState
   状态值(整数)
 
   详细:
      0-未初始化,尚未调用open()方法;
      1-启动,调用了open()方法,未调用send()方法;
      2-发送,已经调用了send()方法,未接收到响应;
      3-接收,已经接收到部分响应数据;
      4-完成,已经接收到全部响应数据;
 
b. Function onreadystatechange
   当readyState的值改变时自动触发执行其对应的函数(回调函数)
 
c. String responseText
   服务器返回的数据(字符串类型)
 
d. XmlDocument responseXML
   服务器返回的数据(Xml对象)
 
e. Number states
   状态码(整数),如:200、404...
 
f. String statesText
   状态文本(字符串),如:OK、NotFound...

2.2 跨浏览器支持

  • XMLHttpRequest
    IE7+, Firefox, Chrome, Opera, etc.
  • ActiveXObject("Microsoft.XMLHTTP")
    IE6, IE5

2.3 实例

native_ajax.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <h3>原生Ajax实例</h3>
    <P><button onclick="XmlGetRequest()">GET发送请求</button></P>
    <P><button onclick="XmlPostRequest()">POST发送请求</button></P>

    <script src="/static/jquery-3.2.1.js"></script>
    <script>
        //该函数可以跨浏览器支持
        function getXhr() {
            var xhr = null;
            if (XMLHttpRequest) {
                xhr = new XMLHttpRequest();
            } else {
                xhr = new ActiveXObject("Microsoft.XMLHTTP");
            }
            return xhr;
        }
        //GET请求
        function XmlGetRequest() {
            var xhr = getXhr();
            //定义回调函数
            xhr.onreadystatechange = function () {
                if(xhr.readyState == 4){
                    // 已经接收到全部响应数据,执行以下操作
                    var response_data1 =xhr.responseText;
                    console.log(response_data1);
                }
            }
            // 指定请求方式、请求地址及是否异步
            xhr.open('GET', "/native_ajax_response/?g1=1&g2=2", true);
            //发送请求
            xhr.send();
        }
        //POST请求
        function XmlPostRequest() {
            var xhr = getXhr();
            //定义回调函数
            xhr.onreadystatechange = function () {
                if(xhr.readyState == 4){
                    // 已经接收到全部响应数据,执行以下操作
                    var response_data2 =xhr.responseText;
                    console.log(response_data2);
                }
            }
            // 指定请求方式、请求地址及是否异步
            xhr.open('POST', "/native_ajax_response/", true);
            // 设置请求头
            xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset-UTF-8');
            //发送请求
            xhr.send('p1=11;p2=22');
        }
    </script>
</body>
</html>

注意:POST请求时,注意请求头:content-type

views.py

def native_ajax(request):
    '''用于原生ajax返回初始页面'''
    if request.method == 'GET':
        return render(request,'native_ajax.html')

def native_ajax_response(request):
    '''用于原生ajax请求响应'''
    if request.method == 'GET':
        g1 = request.GET.get('g1')
        g2 = request.GET.get('g2')
        print(g1,g2)
        return HttpResponse('GET请求成功!')
    else:
        p1 = request.POST.get('p1')
        p2 = request.POST.get('p2')
        print(p1,p2)
        return HttpResponse('POST请求成功!')

执行结果:

image

注意:注释掉CSRF

三 jQuery Ajax

首先我们要知道,jQuery Ajax内部基于“原生Ajax”的。

jQuery其实就是一个JavaScript的类库,其将复杂的功能做了上层封装,使得开发者可以在其基础上写更少的代码实现更多的功能。

  • jQuery 不是生产者,只是是搬运工。
  • jQuery Ajax本质 XMLHttpRequest 或 ActiveXObject

注:2.+版本不再支持IE9以下的浏览器

jQuery.get(...)
	所有参数:
		url: 待载入页面的URL地址
		data: 待发送 Key/value 参数。
		success: 载入成功时回调函数。
		dataType: 返回内容格式,xml, json, script, text, html

jQuery.post(...)
	所有参数:
		url: 待载入页面的URL地址
		data: 待发送 Key/value 参数
		success: 载入成功时回调函数
		dataType: 返回内容格式,xml, json, script, text, html

jQuery.getJSON(...)
	所有参数:
		url: 待载入页面的URL地址
		data: 待发送 Key/value 参数。
		success: 载入成功时回调函数。

jQuery.getScript(...)
	所有参数:
		url: 待载入页面的URL地址
		data: 待发送 Key/value 参数。
		success: 载入成功时回调函数。


jQuery.ajax(...)
	部分参数:
		url:请求地址
		type:请求方式,GET、POST(1.9.0之后用method),默认为GET
		headers:请求头
		data:要发送的数据
		contentType:即将发送信息至服务器的内容编码类型(默认: "application/x-www-form-urlencoded; charset=UTF-8")
		async:是否异步
		timeout:设置请求超时时间(毫秒)

		beforeSend:发送请求前执行的函数(全局)
		complete:完成之后执行的回调函数(全局)
		success:成功之后执行的回调函数(全局)
		error:失败之后执行的回调函数(全局)

		accepts:通过请求头发送给服务器,告诉服务器当前客户端可接受的数据类型
		dataType:将服务器端返回的数据转换成指定类型
			"xml": 将服务器端返回的内容转换成xml格式
			"text": 将服务器端返回的内容转换成普通文本格式
			"html": 将服务器端返回的内容转换成普通文本格式,在插入DOM中时,如果包含JavaScript标签,则会尝试去执行。
			"script": 尝试将返回值当作JavaScript去执行,然后再将服务器端返回的内容转换成普通文本格式
			"json": 将服务器端返回的内容转换成相应的JavaScript对象
			"jsonp": JSONP格式
				     使用 JSONP 形式调用函数时,如 "myurl?callback=?" jQuery 将自动替换 ? 为正确的函数名,以执行回调函数
			如果不指定,jQuery 将自动根据HTTP包MIME信息返回相应类型(an XML MIME type will yield XML, in 1.4 JSON will yield a JavaScript object, in 1.4 script will execute the script, and anything else will be returned as a string

		converters: 转换器,将服务器端的内容根据指定的dataType转换类型,并传值给success回调函数
			$.ajax({
				accepts: {
					mycustomtype: 'application/x-some-custom-type'
				}

				// Expect a `mycustomtype` back from server
				dataType: 'mycustomtype'

				// Instructions for how to deserialize a `mycustomtype`
				converters: {
					'text mycustomtype': function(result) {
					// Do Stuff
					return newresult;
					}
				}
			});
jQuery Ajax 方法列表

 实例

jquery_ajax.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <h3>jQuery Ajax实例--一个简单的计算器</h3>
    <input type="text" id="a1" placeholder="请输入第一个数值">
    +
    <input type="text" id="a2" placeholder="请输入第二个数值">
    =
    <input type="text" id="a3" placeholder="计算结果">
    <input type="button" id="btn" value="jQuery_Ajax计算" onclick="add();" >

    <script src="/static/jquery-3.2.1.js"></script>
    <script>
        function add() {
            $.ajax({
                url:'/jquery_ajax/',
                type:'POST',
                data:{'a1': $('#a1').val(),'a2': $('#a2').val()},
                success:function (arg) {
                    $('#a3').val(arg);
                }
            })
        }
    </script>
</body>
</html>

views.py

def jquery_ajax(request):
    '''用于jQuery Ajax实例'''
    if request.method == 'GET':
        return render(request,'jquery_ajax.html')
    else:
        #这里为了方便,我们之间将其装换为int类型。前端需要输入数字类型
        a1 = int(request.POST.get('a1'))
        a2 = int(request.POST.get('a2'))
        result = a1 + a2
        return HttpResponse(result)

执行结果:

image

四 “伪”Ajax

由于<iframe>标签具有局部加载内容的特性(偷偷的通过HTTP向后台发送数据),所以可以使用其来伪造Ajax请求。

实例一:单独使用iframe(获取指定URL页面)

iframe.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <div>
        <input type="text" id="url" placeholder="请输入需要加载的URL">
        <input type="button" value="刷新" onclick="changeSrc();">
    </div>
    <br>
    <P style="text-align: center">页面加载位置</P>
    <iframe id="ifr1" style="width: 1400px; height: 600px;"></iframe>

    <script>
        function changeSrc() {
            var url = document.getElementById('url').value;
            document.getElementById('ifr1').src = url;
        }
    </script>
</body>
</html>

实例二:混合使用Form+iframe

form_iframe.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <form action="/form_iframe/" method="post" target="ifr1">      <!--target='iframe的name属性'-->
        <iframe name="ifr1" onload="loadIframe()" style="display: none"></iframe>
        <input id='inp1' type="text" name="xxx" />
		<input type="submit" value="提交">
    </form>

    <script>
        function loadIframe() {
            var content = document.getElementById('ifr1').contentWindow.document.body.innerText;
            document.getElementById('inp1').value = content;
            console.log(content);
        }
    </script>
</body>
</html>

views.py

def form_iframe(request):
    '''iframe和form一起使用实现伪造ajax请求'''
    if request.method == "GET":
        return render(request, 'form_iframe.html')
    else:
        print(request.POST.get('xxx'))
        return HttpResponse('iframe返回值')

执行时会报错:

image

form_iframe.html更改如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <form id="f1" action="/form_iframe/" method="post" target="ifr1">      <!--target='iframe的name属性'-->
        <iframe id="ifr1" name="ifr1" style="display: none" ></iframe>
        <input id='inp1' type="text" name="xxx" />
        <button onclick="submitForm();">iframe提交</button>
    </form>

    <script>
        function submitForm(){
            document.getElementById('ifr1').onload = loadIframe;
            document.getElementById('f1').submit();         //找到元素,通过js提交Form数据
        }
        function loadIframe() {
            //获取iframe返回值
            var content = document.getElementById('ifr1').contentWindow.document.body.innerText;
            document.getElementById('inp1').value = content;
        }
    </script>
</body>
</html>

重点注意点:

1.iframe返回值的位置

image

2.iframe返回值的获取方式

var content = document.getElementById('ifr1').contentWindow.document.body.innerText;

3.JS代码提交form表单内容方式

document.getElementById('f1').submit();         //找到元素,通过js提交Form数据

五 CSRF跨站请求伪造

上面的实例都都出现了一个问题,当我们使用’GET‘访问服务端可以正常访问,但使用’POST‘请求访问服务端时,都会报错:403 (Forbidden)

那我们怎么解决呢?主要有以下几种方式:

方式一:

$.ajaxSetup({
	data:{csrfmiddlewaretoken:'{{ csrf_token }}'}
});
注意:要放在ajax请求的前面,在发送之前组装一组字符串,在第一步render的时候就发了
     所以有局限性:如果把JS代码放到静态文件中,不会渲染,不会执行{{csrf_token}},只能在HTML页面中使用

方式二:SCRF放置在请求头中

function add() {
            $.ajax({
                url:'/jquery_ajax/',
                type:'POST',
                data:{'a1': $('#a1').val(),'a2': $('#a2').val()},
                headers:{"X-CSRFToken":$.cookie('csrftoken')},    //设置头信息
                success:function (arg) {
                    $('#a3').val(arg);
                }
            })
        }

前提是我们需要先导入一个有关cookie操作的一个插件:

<script src="/static/jquery.cookie.js"></script>

方式三:SCRF放置在data中

function add() {
	$.ajax({
		url:'/jquery_ajax/',
		type:'POST',
		data:{……,'csrfmiddlewaretoken': $('input[name="csrfmiddlewaretoken"]').val()},
		success:function (arg) {
			$('#a3').val(arg);
		}
	})
}

前提是我们需要在body中加入{% csrf_token %},先获取到CSRF的值,后面才能获取其值发送给服务端。

 

以上方式二、三在之前的文章:WEB框架--DJANGO之XSS&CSRF有详细例子。

posted @ 2018-11-02 18:06  Joe1991  阅读(108)  评论(0)    收藏  举报