Django与Ajax


Django与Ajax

AJAX准备知识:JSON

什么是 JSON ?

  • JSON 指的是 JavaScript 对象表示法(JavaScript Object Notation)
  • JSON 是轻量级的文本数据交换格式
  • JSON 独立于语言 *
  • JSON 具有自我描述性,更易理解

* JSON 使用 JavaScript 语法来描述数据对象,但是 JSON 仍然独立于语言和平台。JSON 解析器和 JSON 库支持许多不同的编程语言。

啥都别多说了,上图吧!

合格的json对象(json只认双引的字符串格式):

["one", "two", "three"] [ , ] 

不合格的json对象:

// 属性名必须使用双引号 [32, 64, 128, 0xFFF] // 不能使用十六进制值 // 不能使用undefined // 不能使用函数和日期对象 }

stringify与parse方法

JavaScript中关于JSON对象和字符串转换的两个方法:

JSON.parse(): 用于将一个 JSON 字符串转换为 JavaScript 对象(json只认双引的字符串格式)

JSON.parse(''); JSON.parse('') ; // 错误 JSON.parse('[18,undefined]') ; // 错误

JSON.stringify(): 用于将 JavaScript 值转换为 JSON 字符串。

JSON.stringify()

和XML的比较

JSON 格式于2001年由 Douglas Crockford 提出,目的就是取代繁琐笨重的 XML 格式。

JSON 格式有两个显著的优点:书写简单,一目了然;符合 JavaScript 原生语法,可以由解释引擎直接处理,不用另外添加解析代码。所以,JSON迅速被接受,已经成为各大网站交换数据的标准格式,并被写入ECMAScript 5,成为标准的一部分。

XML和JSON都使用结构化方法来标记数据,下面来做一个简单的比较。

用XML表示中国部分省市数据如下:

<?xml version="1.0" encoding="utf8"?> <country> <name>中国</name> <province> <name>黑龙江</name> <cities> <city>哈尔滨</city> <city>大庆</city> </cities> </province> <province> <name>广东</name> <cities> <city>广州</city> <city>深圳</city> <city>珠海</city> </cities> </province> <province> <name>台湾</name> <cities> <city>台北</city> <city>高雄</city> </cities> </province> <province> <name>新疆</name> <cities> <city>乌鲁木齐</city> </cities> </province> </country>

用JSON表示如下:

}, }, }, }] }

由上面的两端代码可以看出,JSON 简单的语法格式和清晰的层次结构明显要比 XML 容易阅读,并且在数据交换方面,由于 JSON 所使用的字符要比 XML 少得多,可以大大得节约传输数据所占用得带宽

什么是Ajax

AJAX(Asynchronous Javascript And XML)翻译成中文就是“异步Javascript和XML”。即使用Javascript语言与服务器进行异步交互,传输的数据为XML(当然,传输的数据不只是XML,现在更多使用json数据)。

  • 同步交互:客户端发出一个请求后,需要等待服务器响应结束后,才能发出第二个请求;
  • 异步交互:客户端发出一个请求后,无需等待服务器响应结束,就可以发出第二个请求。
  • 因为服务器响应内容不再是整个页面,而是页面中的部分内容,所以AJAX性能高;
  • 两个关键点:1.局部刷新,2.异步请求

AJAX除了异步的特点外,还有一个就是:浏览器页面局部刷新;(这一特点给用户的感受是在不知不觉中完成请求和响应过程)

优点:

  • AJAX使用Javascript技术向服务器发送异步请求
  • AJAX无须刷新整个页面

示例

页面输入两个整数,通过AJAX传输到后端计算出结果并返回。

HTML

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF8"> <title>Title</title> <meta name="viewport" content="width=devicewidth, initialscale=1"> </head> <body> <input type="text" id="i1"> + <input type="text" id="i2"> = <input type="text" id="i3"> <button id="b1">Ajax Test</button> <script src="/static/jquery3.3.1.min.js"></script> <script> $('#b1')lick(function () , success:function (data) }) }) </script> </body> </html>

views.py

def ajax_test(request): if request.method=='POST': i1=request.POST.get('i1') i2=request.POST.get('i2') ret=int(i1)+int(i2) return HttpResponse(ret) return render(request,'ajax_test')

urls.py

fr djangoonf.urls import url fr app01 import views urlpatterns=[ url(r'^ajax_test/',views.ajax_test), ]

AJAX常见应用情景

搜索引擎根据用户输入的关键字,自动提示检索关键字。

还有一个很重要的应用场景就是注册时候的用户名的查重。

其实这里就使用了AJAX技术!当文件框发生了输入变化时,使用AJAX技术向服务器发送一个请求,然后服务器会把查询到的结果响应给浏览器,最后再把后端返回的结果展示出来。

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

基于jquery的Ajax实现

最基本的jQuery发送AJAX请求示例:

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF8"> <title>Title</title> <meta name="viewport" content="width=devicewidth, initialscale=1"> <style> .hide </style> </head> <body> <p><input type="text" class="user"><span class="hide" style="color: red">用户名已存在</span></p> <script src="/static/jquery3.3.1.min.js"></script> <script src="/static/setup_Ajax.js"></script> <script> //给input框绑定一个失去焦点的事件 $('.user').blur(function () , //success为回调函数,参数data即后端给你返回的数据 success:function (data) } }) }); </script> </body> </html>

views.py:

def index(request): if request.method=='POST': ret= username=request.POST.get('username') if username=='JBY': ret['flag']=True import json return HttpResponse(json.dumps(ret)) return render(request,'index')

$.ajax参数

data参数中的键值对,如果值值不为字符串,需要将其转换成字符串类型。

$("#b1").on("click", function () , success:function (data) }) })

通过Ajax,实现前端输入两个数字,服务器做加法,返回到前端页面

视图函数

def test_ajax(requests): n1=int(requests.POST.get('n1')) n2=int(requests.POST.get('n2')) return HttpResponse(n1+n2)

js代码

$("#submit")lick(function () , success: function (data) , }) })

Html代码

<input type="text" id="num1">+<input type="text" id="num2">=<input type="text" id="sum"> <button id="submit">计算</button>

二 基于Ajax进行登录验证

用户在表单输入用户名与密码,通过Ajax提交给服务器,服务器验证后返回响应信息,客户端通过响应信息确定是否登录成功,成功,则跳转到首页,否则,在页面上显示相应的错误信息

视图函数

def auth(request): back_dic= name=request.POST.get('user') password=request.POST.get('password') print(name) print(password) user=models.user.objects.filter(name=name,password=password).first() print(user) # print(user.query) if user: back_dic['user']=user.name back_dic['message']='成功' else: back_dic['message']='用户名或密码错误' import json return HttpResponse(json.dumps(back_dic))

Js代码

$("#submit3")lick(function () , success: function (data) var data=JSON.parse(data) if (data.user)else ) } } }) } )

traditional:true>可以序列化一层列表,多层不行,要转成json格式上传

JS实现AJAX(了解)

var b2 = document.getElementById("b2"); b2.onclick = function () }; };

AJAX请求如何设置csrf_token

不论是ajax还是谁,只要是向我Django提交post请求的数据,都必须校验csrf_token来防伪跨站请求,那么如何在我的ajax中弄这个csrf_token呢,我又不像form表单那样可以在表单内部通过一句就搞定了......

方式1

通过获取隐藏的input标签中的csrfmiddlewaretoken值,放置在data中发送。

$.ajax(, success: function (data) })

方式2

通过获取返回的cookie中的字符串 放置在请求头中发送。

注意:需要引入一个jqueryookie.js插件。

$.ajax(, // 从Cookie取csrf_token,并设置ajax请求头 data: , success: function (data) })

方式3

或者用自己写一个getCookie方法:

function getCookie(name) } } return cookieValue; } var csrftoken = getCookie('csrftoken');

每一次都这么写太麻烦了,可以使用$.ajaxSetup()方法为ajax请求统一设置。

function csrfSafeMethod(method) $.ajaxSetup( } });

将下面的文件配置到你的Django项目的静态文件中,在html页面上通过导入该文件即可自动帮我们解决ajax提交post数据时校验csrf_token的问题,(导入该配置文件之前,需要先导入jQuery,因为这个配置文件内的内容是基于jQuery来实现的)

更多细节详见:Djagno官方文档中关于CSRF的内容

文件上传

请求头ContentType

1 application/xwwwformurlencoded

这应该是最常见的 POST 提交数据的方式了。浏览器的原生

表单,如果不设置

POST example HTTP/1.1 ContentType: application/xwwwformurlencoded;charset=utf8 user=lqz&age=22

2 multipart/formdata

这又是一个常见的 POST 数据提交的方式。我们使用表单上传文件时,必须让 表单的 enctype 等于 multipart/formdata。直接来看一个请求示例:

POST example HTTP/1.1 ContentType:multipart/formdata; boundary=WebKitFormBoundaryrGKCBY7qhFd3TrwA WebKitFormBoundaryrGKCBY7qhFd3TrwA ContentDisposition: formdata; name="user" yuan WebKitFormBoundaryrGKCBY7qhFd3TrwA ContentDisposition: formdata; name="file"; filename="chre.png" ContentType: image/png PNG ... content of chre.png ... WebKitFormBoundaryrGKCBY7qhFd3TrwA

这个例子稍微复杂点。首先生成了一个 boundary 用于分割不同的字段,为了避免与正文内容重复,boundary 很长很复杂。然后 ContentType 里指明了数据是以 multipart/formdata 来编码,本次请求的 boundary 是什么内容。消息主体里按照字段个数又分为多个结构类似的部分,每部分都是以 boundary 开始,紧接着是内容描述信息,然后是回车,最后是字段具体内容(文本或二进制)。如果传输的是文件,还要包含文件名和文件类型信息。消息主体最后以 boundary 标示结束。关于 multipart/formdata 的详细定义,请前往 rfc1867 查看。

这种方式一般用来上传文件,各大服务端语言对它也有着良好的支持。

上面提到的这两种 POST 数据的方式,都是浏览器原生支持的,而且现阶段标准中原生 表单也只支持这两种方式(通过 元素的 enctype 属性指定,默认为 application/xwwwformurlencoded。其实 enctype 还支持 text/plain,不过用得非常少)。

随着越来越多的 Web 站点,尤其是 WebApp,全部使用 Ajax 进行数据交互之后,我们完全可以定义新的数据提交方式,给开发带来更多便利。

3 application/json

application/json 这个 ContentType 作为响应头大家肯定不陌生。实际上,现在越来越多的人把它作为请求头,用来告诉服务端消息主体是序列化后的 JSON 字符串。由于 JSON 规范的流行,除了低版本 IE 之外的各大浏览器都原生支持 JSON.stringify,服务端语言也都有处理 JSON 的函数,使用 JSON 不会遇上什么麻烦。

JSON 格式支持比键值对复杂得多的结构化数据,这一点也很有用。记得我几年前做一个项目时,需要提交的数据层次非常深,我就是把数据 JSON 序列化之后来提交的。不过当时我是把 JSON 字符串作为 val,仍然放在键值对里,以 xwwwformurlencoded 方式提交。

基于Form表单上传文件

<form action="/file_put/" method="post" enctype="multipart/formdata"> 用户名:<input type="text" name="name"> 头像:<input type="file" name="avatar" id="avatar1"> <input type="submit" value="提交"> </form>

必须指定 enctype="multipart/formdata"

视图函数:

def file_put(request): if request.method=='GET': return render(request,'file_put') else: # print(request.POST) # print(request.POST) print(request.body) # 原始的请求体数据 print(request.GET) # GET请求数据 print(request.POST) # POST请求数据 print(request.FILES) # 上传的文件数据 # print(request.body.decode('utf8')) print(request.body.decode('utf8')) print(request.FILES) file_obj=request.FILES.get('avatar') print(type(file_obj)) with open(file_obj.name,'wb') as f: for line in file_obj: f.write(line) return HttpResponse('ok')

基于Ajax上传文件

$("#ajax_button")lick(function () }) })

浏览器请求头为:

ContentType:

multipart/formdata; boundary=WebKitFormBoundaryA5O53SvUXJaF11O2

Ajax提交json格式数据

$("#ajax_test")lick(function () $.ajax( }) })

提交到服务器的数据都在 request.body 里,取出来自行处理

Django内置的serializers(把对象序列化成json字符串)

什么意思呢?就是我的前段想拿到由ORM得到的数据库里面的一个个用户对象,我的后端想直接将实例化出来的数据对象直接发送给客户端,那么这个时候,就可以用Django给我们提供的序列化方式

fr djangoore import serializers fr djangoore import serializers def test(request): book_list = Book.objects.all() ret = serializers.serialize("json", book_list) return HttpResponse(ret)

补充一个SweetAlert插件示例

点击下载Bootstrapsweetalert项目。

$("#b55")lick(function () , function () , success: function (data) else } }) }); })

上面这个二次确认的动态框样式,你也可以直接应用到你的项目中

提醒事项:

1.上述的样式类部分渲染的样式来自于bootstrap中,所有建议在使用上述样式时,将bootstrap的js和css也导入了,这样的情况下,页面效果就不会有任何问题

2.弹出的上述模态框中,可能字体会被图标掩盖一部分,可通过调整字体的上外边距来解决



上一篇:BOM

下一篇:MySQL 库、表、记录、相关操作(1)


Copyright © 2002-2019 k262电脑网 www.k262.cn 皖ICP备2020016292号
温馨提示:部分文章图片数据来源与网络,仅供参考!版权归原作者所有,如有侵权请联系删除!QQ:251442993 热门搜索 网站地图