Ajax基础(应用场景|jquery实现Ajax|注意事项|Ajax发送json数据|Ajax携带文件数据)_jquery ajax-程序员宅基地

技术标签: Django  ajax  jquery  

一、Ajax简介

AJAX(Asynchronous Javascript And XML)翻译成中文就是“异步Javascript和XML”。即使用Javascript语言与服务器进行异步交互也是用XML进行数据传输(当然,传输的数据不只是XML,目前更多的使用Json数据)。
(XML应用于Web开发的许多方面,常用于简化数据的存储和共享)

	AJAX不是什么新的编程语言,而是一种使用现有标准的新方法
	AJAX最大的优点就是不需要重新加载整个页面的情况下,可以与服务器交换数据并更新网页内容
	(简单来说就是,在网页前端刚输入完数据,后端其实已经验证完数据了,不需要等点击提交了)
	AJAX不需要任何浏览器插件,但需要用户运行Javascript在浏览器上执行

	1.AJAX使用Javascript技术向服务器发公司异步请求
	2.AJAX请求无需刷新整个页面
	3.因为服务器响应内容不再是整个页面,而是页面中的部分内容,所以AjAX性能高。

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

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


常见的使用场景
在这里插入图片描述

注册用户时,会根据用户输入的命令关键字,自动提示改关键字是否通过检验

很多网站注册的时候都会使用到AJAX技术,当文本框发生了输入变化时,使用AJAX技术向服务器发送一个请求,然后服务器会把查询到的结果响应到的结果响应给浏览器,期间浏览器是不用刷新的,最后把后端返回的结果展示出来。

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

二、基于jquery实现Ajax

目前Ajax一般不会使用原生Javascript来编写,因为需要考虑不同浏览器的兼容性。我们这里通过jquery来实现,更简单、不需要考虑不同浏览器引发的兼容问题。

过程:我们通过在前端向一个URL发送Ajax请求,来让后端处理这个请求后返回数据给Ajax接收。

这里用一个案例来代入:
通过Ajax,实现前端输入两个数字,服务器做加法,返回到前端页面

先在Django配置路由

	from django.conf.urls import url
	from django.contrib import admin
	from app import views
	
	urlpatterns = [
	    url(r'^admin/', admin.site.urls),
	    url(r'^index/', views.index),
	    url(r'^sum/', views.sums),  # ajax会向这个路由提交请求,执行sums视图函数
	]

注意:一定要在settings.py配置文件里面注释中间件的一行内容
在这里插入图片描述
这行代码的作用后续在讲解,目前先注释掉,不然请求会提交不成功!


views.py处理请求

	from django.shortcuts import render,HttpResponse
	
	def index(request):  # 返回一个index页面
    	return render(request,'index.html')

	def sums(request):  # 处理ajax请求
	    num1 = int(request.POST.get('num1'))
	    num2 = int(request.POST.get('num2'))
	    return HttpResponse(num1+num2)  # 返回计算后的值

index.html文件,定义Ajax与后端进行交互

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.7.1/jquery.min.js"></script>
    <link href="https://cdn.bootcdn.net/ajax/libs/twitter-bootstrap/3.4.1/css/bootstrap.min.css" rel="stylesheet">
    <script src="https://cdn.bootcdn.net/ajax/libs/twitter-bootstrap/3.4.1/js/bootstrap.min.js"></script>
    <style>
        *{
    
            margin-top:20px;
            margin-left:20px;
        }
    </style>
</head>
<body>
<input type="text" id="num1" placeholder="数字1">+
<input type="text" id="num2" placeholder="数字2">=
<input type="text" id="num3" placeholder="总和">

<button id="btn">计算</button>


<script>
    $('#btn').click(function (){
      // 当按钮被点击时触发一个匿名函数
        var num1 = $('#num1').val()  // 点击按钮后获取id属性值为num1的值
        var num2 = $('#num2').val()  // 点击按钮后获取id属性值为num2的值
        $.ajax({
      // 定义ajax发送请求
            url:'/sum/',  // 请求发送的地址  有三种填写方式,与form标签的action一致
            method:'post',  // 请求发送的方式
            data: {
    'num1':num1,'num2':num2},  // 请求携带的数据
            success:function(data){
      // 异步等待,当后端响应成功会回调执行匿名函数,并将数据传递给data参数
                $('#num3').val(data)  // 将数据设置给id属性值为num3的文本框
            }// 使用ajax交互,那么后端返回的数据会被data接收,不在直接影响整个浏览器页面

        })
    })
</script>
</body>
</html>

我们已经编写好了一个简单的小案例,通过Ajax向后端发送请求,后端处理完数据响应给Ajax,再将得到的数据在页面使用以此来达到局部更新页面效果。而form表单要达到这一效果需要全局更新数据,使用重定向来实现。

得到的结果:
在这里插入图片描述


三、使用Ajax注意的问题

1.Ajax不要与form表单同时提交

  • 如果在form表单中,写了button按钮或者input是submit类型的话,点击会触发form表单的提交。
  • 如果点击按钮既触发Ajax又触发了form表单的提交,那么就会发送两个请求,导致页面数据可能错乱或者不显示问题。
  • 要是无可避免同时使用Ajax提交form表单下的表单控件时,form表单内的input按钮请使用button类型,这样该按钮绑定了Ajax请求的话,就只会提交Ajax请求,而不是form
	<!--这种类型的按钮,无法触发form表单的提交-->
	<input type="button" value="提交">
	
	<!--如果该button在form内,则会触发form表单的提交-->
	<button>提交</button>

2.后端响应格式问题

  • 后端如果是通过非JsonResponse返回值的话,响应格式为text/html,前端得到数据后需要手动转换成对象的形式。
    if request.is_ajax():
        import json
        d = {
    'name':'jack','age':18}
        d_json = json.dumps(d,ensure_ascii=False)
        return HttpResponse(d_json)

就算我们在后端已经将字典转换成了Json格式,但是通过HttpResponse返回以后,还是变成了一堆字符串到前端,并且响应格式为:text/html

<button id="btn">点击发送Ajax请求</button>
<script>
   $('#btn').click(function(){
    
       $.ajax({
    
           url:'/home/',
           method:'post',
           data:{
    'name':'tom'},
           success:function (data){
    
               console.log(data)
               console.log(data.name)  // 如果返回的是Json的话这样是可以打印出来的
           }
       })
   })
</script>

查看后端响应的数据格式:

在这里插入图片描述
控制台结果:
在这里插入图片描述
需要在前端手动将数据转换成JSON格式,那么下次就会先将数据转换成JSON,然后再打印了

	success:function (data){
    
		data = JSON.parse(data)
		console.log(data)
		console.log(data.name)
	}
  • 如果我们是通过JsonResponse返回的话,那么响应状态码就会是application/json,并且在Ajax内接收到的也会是一个JSON格式的数据
	from django.http import JsonResponse
	def home(request):
	    if request.is_ajax():
	        import json
	        d = {
    'name':'jack','age':18}
	        return JsonResponse(d)
	    return render(request,'home.html')

Ajax内接收到以后不需要手动转换成JSON格式

	success:function (data){
    
	    console.log(data)
	    console.log(data.name)
   }

查看响应头编码格式
在这里插入图片描述
控制台打印效果:
在这里插入图片描述

总结:

  1. 后端如果使用retrun JsonResponse(d),前端不能反序列化,直接当成对象使用
  2. 后端使用return HttpResponse(d),前端需要自己反序列化:data=JSON.parse(data)
  3. 后端使用return HttpResponse(d),前端不反序列化,需要指定参数:datatype:‘json’

3、使用了Ajax作为请求后的注意事项

  • 后端不要返回render、redirect、HttpResponse。因为这些内容会被统一当做字符串返回给Ajax
	def test(request):
    if request.is_ajax():
        return render(request,'login.html')

    return render(request,'test.html')

控制台打印效果:
在这里插入图片描述

总结:使用了AJax作为请求以后,建议使用JsonResponse返回数据,虽说最终还是通过HttpResponse返回到web的,但是JsonResponse内部就做了响应格式的转换

在这里插入图片描述

这种和我们直接通过HttpResponse返回一个JSON格式数据是不一样的。AJax只会根据响应码来做处理


四、前后端数据传输的编码格式(content-Type)

我们前后端数据交互格式其实有很多种(普通键值对、文件、JSON格式)每一种格式发过去之后,它是怎么样动态识别的呢?

我们通过网页右键检查>Network>Content-Type这个数据表示当前数据是什么编码,可以通过form表单中的enctype去修改类型
针对不同的格式数据Django后端会有不同的响应

Form-data所有的数据都会被解析到reqeust.files里面,二进制是无法查阅的
Urlencoded会被解析到request.post里面
application/json它的数据格式是json格式,这个格式form表单发送不了,只有Ajax可以发送

1.urlencoded

	Ajax默认的编码格式、form表单默认的编码格式
	数据格式:xxx=yyy&uuu=ooo&aaa=bbb
	django后端会自动处理到request.POST中

2.formdata

formdata就是form表单的第三个参数enctype=‘multipart/form-data’,它会使浏览器接收到的数据格式更改,使之post请求方式无法获取到提交到后端的数据

django后端针对普通的键值对还是处理到request.POST中,但是针对文件会处理到request.Field

3.application/json

	'''模拟Ajax发送数据'''

	'html代码'
	<button id="d1">点击我发送数据</button>
	<script>
		$('#d1').click(function () {
    
	        $.ajax({
    
	            url:'',
	            type:'post',
	            data:JSON.stringify({
    'name':'jack', 'age':18}),    // 必须json格式
	            contentType:'application/json', // 要制定模式 默认Urlencoded
	            success:function (data) {
    
	                {
    #console.log(typeof data)#}
	            }
	        })
	    })
    </script>

	views.py
	def index(request):
    if request.is_ajax():  # 判断当前是不是ajax请求专门的方法
        if request.method == 'POST':
            print(request.POST)  # <QueryDict: {}>
            print(request.FILES)  # <MultiValueDict: {}>
            '''Django后端针对json格式数据处理不了 会放在request.body里面'''
            print(request.body)  # b'{"name":"jack","age":18}'
            '''Django它不会去处理需要我们自己去处理'''
            json_bytes = request.body
            import json
            json_dict = json.loads(json_bytes)  # Json解码
            print(json_dict, type(json_dict))

    return render(request,'index.html')

form表单只支持urlencodedformdata两个数据格式,但ajax支持三种数据格式(urlencoded、formdata和application/json)。


五、Ajax携带文件数据

在form表单内,我们是通过指定了一下编码格式才可以将文件上传到后端,而通过Ajax我们将不再借助form表单来实现这一效果

	'html代码'
	
	<input type="text" id="ts1" >
	<input type="file" id="myfile" >
	<button id="btn">点我发送数据</button>
	
	<script>
	    $('#btn').click(function(){
    
	        //1.产生内置对象
	        //实例化一个对象,formdata对象用来保存key/value结构的数据,通常用于form传输数据
	        var MyFormDataObj = new FormData;
	
	        var ts1 = $('#ts1').val();
	        //$('#myfile')[0]获取原生JS、$('#myfile')[0].files获取用户上传的所有文件
	        var myfile = $('#myfile')[0].files[0];
	        {
    #console.log(myfile)#}
	        //2.添加普通数据
	        MyFormDataObj.append('ts1',ts1)
	        //3.添加文件数据
	        MyFormDataObj.append('myfile',myfile) //给该文件定义一个取值名称,第一个参数自定义,第二个参数:值
	        //4.发送Ajax请求
	        $.ajax({
    
	            url:'',
	            type:'post',
	            data:MyFormDataObj,  //本身含有编码,无需做声明
	            contentType:false,   //不使用任何编码,必须添加固定格式
	            processData: false,  //不处理数据对象
	            success:function(data){
    
	            	console.log(data);
	                console.log(data.msg);
	            }
	        })
	    })
	</script>
	
	'后端views.py'
	def index(request):
    if request.is_ajax():  # 判断当前是不是ajax请求专门的方法
        if request.method == 'POST':
            file = request.FILES.get('myfile')
            import os
            DIR_PATH = os.path.dirname(__file__)

            with open(DIR_PATH+file.name, 'wb')as f:
                for i in file:
                    f.write(i)

            return JsonResponse({
    'status': 100, 'msg': '上传成功'})

    return render(request,'index.html')

最终结果:
在这里插入图片描述

后端还是通过这两个请求方法获取文件和普通数据,Ajax后端也是一样的。只是前端不一样需要添加contentTypeprocessdata参数

如果我们要实现多个文件上传,稍微做出一些改动即可:

	'html代码'
	<input type="file" id="myfile" multiple> <!--允许上传多个文件-->
	<button id="btn">上传文件</button>
	
	<script>
	    $('#btn').click(function(){
    
	        //实例化一个对象用来保存key/value结构的数据,通常用于form传输数据
	        var FormDataObj = new FormData;
	        var myfile = $('#myfile')[0].files; //获取需要上传的所有文件,拿到的是一个列表的形式
	
	        for(var i =0;i<myfile.length;i++){
     //根据上传文件的数量来进行遍历
	            FormDataObj.append(myfile[i].name,myfile[i])
	            //将每一个文件的名称作为key,文件作为value追加到FormData对象内
	        }
	
	        $.ajax({
    
	            url:'/index/',
	            type:'post',
	            contentType:false,
	            processData:false,
	            data:FormDataObj,
	            success:function(data){
    
	                console.log(data)
	            }
	        })
	
	    })
	<script>

	'后端views.py'
	from django.http import JsonResponse
	def index(request):
	    if request.is_ajax():  # 判断当前是不是ajax请求专门的方法
	        import os
	        DIR_PATH = os.path.dirname(__file__)
	        file = request.FILES
	        print(file)
	
	        for k in file:
	            fl1 = file.get(k)
	            with open(DIR_PATH + fl1.name, 'wb') as f:
	                for i in fl1:
	                    f.write(i)
	
	        return JsonResponse({
    'status': 200, 'msg': '上传成功'})
	
	    return render(request,'index.html')
	

最终结果:
在这里插入图片描述


六、Ajax发送json格式数据

通过在Ajax内指定好编码格式,然后将js的数据类型转换成JSON格式数据上传给后端

	1.确保发送json对应的数据是json格式字符串
		data:JSON.stringfly({
    })
	2.修改数据编码格式(默认是urlencoded)
		contentType:'application/json'

home.html

	<p>用户名: <input type="text" id="username" ></p>
	<p>密码: <input type="password" id="password" ></p>
	<button id="btn">提交</button>	
	<script>
	    $('#btn').click(function (){
    
	        var name = $('#username').val()
	        var password = $('#password').val()
	
	        $.ajax({
    
	            url:'/home/',
	            type:'post',
	            data:JSON.stringify({
    'username':name,'password':password}),
	            contentType:'application/json',
	            success:function (data){
    
	                console.log(data)
	            }
	        })
	    })
	</script>

后端views.py:并不能同以往来处理请求里面的数据了

	import json
	def home(request):
	    if request.is_ajax():
	        if request.method == 'POST':
	            # json格式,从POST中取不出来
	            name = request.POST.get('username')
	            # print(type(request.POST)) #<class 'django.http.request.QueryDict'>
	            # print(name)  # None
	
	            # Ajax上传的JSON格式数据在request.body中,且是bytes类型
	            print(request.body)  # b'{"username":"fdsa","password":"fdsa"}
	
	            #在python3.6之后loads可以将bytes里面如果有JSON格式数据,可以转换出来
	            request.data = json.loads(request.body)
	            name = request.data.get('username')
	            password = request.data.get('password')
	            print(name,password) # 123 456
	            return JsonResponse({
    'status':200,'msg':'提交成功'})
	
	    return render(request,'home.html')

最终结果:
在这里插入图片描述


版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/achen_m/article/details/134513175

智能推荐

[Android Exercise]仿微信游戏界面PART.1—ConstraintLayout和RecyclerView的应用_android微信游戏界面-程序员宅基地

文章浏览阅读860次。1、分析模块将每一个部分都做成一个_android微信游戏界面

关于vue-cli3的浏览器兼容性_@vue/babel-preset-app-程序员宅基地

文章浏览阅读1.4w次,点赞3次,收藏17次。这里先给出几个链接:1.Vue-cli浏览器兼容性:https://cli.vuejs.org/guide/browser-compatibility.html#usebuiltins-usage2.browserslist:https://www.npmjs.com/package/browserslist3.babel.config.js:https://babeljs.io/..._@vue/babel-preset-app

nginx负载均衡 upstream ip_hash的用法_upstream-hash-by: "$remote_ip-程序员宅基地

文章浏览阅读7.2k次。文章目录场景参考文档用法场景负载均衡解决session共享的问题参考文档nginx.org upstream用法语法Syntax: ip_hash;Default: —Context: upstream说明Specifies that a group should use a load balancing method where requests are d..._upstream-hash-by: "$remote_ip

自适应滤波算法(LMS算法)-程序员宅基地

文章浏览阅读9.7k次,点赞6次,收藏68次。引言 LMS学习算法是由Widrow和Hoff于1960年提出的,该算法也称为Δ\DeltaΔ规则,该算法与感知器网络的学习算法在权值调整上都基于纠错学习规则,但LMS算法那更容易实现,因此得到了广泛应用。   注意:LMS算法只能训练单层网络,但这并不影响其功能,从理论上讲,多层线性网络并不比单层网络强大,它们具有同样的能力,即对于每一个多层线性网络,都具有一个等效的单层..._lms算法

【文末福利】为什么我们要掌握Linux系统编程?_学习了linux系统编程可以结合什么用-程序员宅基地

文章浏览阅读9.1k次,点赞9次,收藏14次。作为一个嵌入式开发者,我觉得基于Linux的系统编程,这个应该是绕不开的话题。本文将围绕,为何要掌握Linux系统编程这个问题,给出一些观点,希望对各位有所帮助。_学习了linux系统编程可以结合什么用

LoRaAN终端OTAA入网方式的详细介绍_join_devnonce_err-程序员宅基地

文章浏览阅读1k次。在之前的文章中向大家介绍了“该如何选择LoRaWAN终端入的网方式”,本文主要介绍了OTAA节点是如何入网的。此文来自微信公众号“小七说LoRa”,内容已获小七老师授权,小七老师是腾讯云在线课程讲师,点击链接https://mp.weixin.qq.com/s/WvGmi81zBMzbAX8OUb-DpA可以观看课程视频。OTAA的全称是Over The Air Activation。它的入网步骤是这样的:节点发出的Join Request请求通过网关转发到服务器,也就是NS;NS会对该请求做一些判断处_join_devnonce_err

随便推点

【java】根据当前时区获取时间_java 根据asia/shanghai 换算时间简称-程序员宅基地

文章浏览阅读3.7k次。地区国家 编号 缩写 时区中国 86 CN Asia/Shanghai香港 852 HK Asia/Hong_Kong澳门 853 MO Asia/Macau台湾 886 TW Asia/Taipei新加坡 65 SG Asia/Singapore泰国 66 TH Asia/Bangkok印度 91 IN Asia/Calcutta日本 81 JP Asia/Tokyo韩国 82 KR Asia/Seoul巴基斯坦 92 PK Asia/Karachi美国 1 US America/._java 根据asia/shanghai 换算时间简称

单精度浮点数和双精度浮点数存储_双精度浮点型存储和单精度浮点型存储-程序员宅基地

文章浏览阅读694次,点赞3次,收藏2次。简要对IEEE 754中 单精度浮点数和双精度浮点数存储进行一些分析和解释,欢迎大家来补充纠正_双精度浮点型存储和单精度浮点型存储

Android屏幕适配-重点盘点_android 屏幕适配-程序员宅基地

文章浏览阅读1.1k次。享学课堂诚邀作者:周周转载请声明出处!引子屏幕适配是 android 开发/面试 绕不开的一个问题。本文 将屏幕适配的知识要点完整展现给各位读者。正文大纲android需要做屏幕适配的原因基础知识点(很重要)屏幕适配攻略正文android需要做屏幕适配的原因关键字:android碎片化android面世以来,google开源了android系统,各家厂商各自为政,导致屏幕尺寸没有统一标准,屏幕的宽高比各种各样,屏幕密度也是各个厂家攀比的资本, 导致Android开发者想要._android 屏幕适配

awvs 中文手册详细版-程序员宅基地

文章浏览阅读119次。awvs 中文手册详细版目录:0×00、什么是Acunetix Web Vulnarability Scanner ( What is AWVS?)0×01、AWVS安装过程、主要文件介绍、界面简介、主要操作区域简介(Install AWVS and GUI Description)0×02、AWVS的菜单栏、工具栏简介(AWVS menu bar & too..._awvs automatic login failed for 10.68.120.203

“开启IT管理新时代”惠普软件客户论坛圆满闭幕-程序员宅基地

文章浏览阅读74次。2012年3月8日,由惠普软件主办的“开启IT管理新时代”客户论坛在北京富力万丽酒店盛大召开,来自政府、金融、电信、制造行业的众多IT主管代表出席了本论坛。本次论坛分设运维管理和应用生命周期管理两个分论坛。现场,参会嘉宾了解到了惠普最新重金投入研发的惠普IT绩效管理解决方案,以及该方案在帮助IT主管更好地实现全面IT运维管理、量化IT衡量指标方面的特点和优势。告别“千人一..._it公司 技术总监 技术论坛闭幕

Android系统属性值设置和使用_property_value_max-程序员宅基地

文章浏览阅读2.8k次。转载:https://blog.csdn.net/qq_30624591/article/details/102679377每个属性都有一个名称和值,他们都是字符串格式。属性被大量使用在Android系统中,用来记录系统设置或进程之间的信息交换。属性是在整个系统中全局可见的。每个进程可以get/set属性。属性服务在“init”守护进程中运行。每一个客户端想要设置属性时,必须连接属性服务..._property_value_max

推荐文章

热门文章

相关标签