Django Ajax

Ajax

1    大坑
    如果在form表单中,写button和input是submit类型,会触发form表单的提交
    如果不想触发
        不写在form表单里面
        将input的type写成button
   
2   大坑
    后端响应格式如果是:html/text格式,ajax接收到的数据后需要自己转成对象
    总结:后端返回数据,统一都用JsonResponse
    
3   大坑
    如果使用了ajax,后端就不要返回rediret,render,HttpResponse
    直接返回JsonResponse

 

详情参见:https://www.cnblogs.com/Dominic-Ji/p/9234099.html

"""
异步提交:请求发出去 不会卡在这里,可以干其他事情
局部刷新:js的DOM操作 使页面内局部刷新

基本上很多web页面都有很多ajax请求
例子:github注册
    动态获取用户名,实时跟后端确认,并实时展示到前端(局部刷新)
    
朝后端发送请求的方式
    1、浏览器地址栏直接输入url回车      GET请求
    2、a标签href属性                 GET请求
    3、form表单                      GET请求/POST请求
    4、ajax                            GET请求/POST请求
    

AJAX 不是新的编程语言,而是一种使用现有标准的新方法。
    就类似于装饰器 基于现有的知识点拼凑出来的
    
    
AJAX 最大的优点是在不重新加载整个页面的情况下,可以与服务器交换数据并更新部分网页内容。(这一特点给用户的感受是在不知不觉中完成请求和响应过程)



AJAX 不需要任何浏览器插件,但需要用户允许JavaScript在浏览器上执行。


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


Ajax只学习jQuery封装之后的版本(不学原生的 原生的复杂并且在实际项目中也一般不用)
所以我们在前端页面使用ajax的时候需要确保导入了jQuery
ps:并不只有jQuery能够实现ajax,其他框架也可以,但是换汤不换药 原理是一样的
"""

Ajax基本语法

"""
$.ajax({
    url:'',  # 朝后端哪个地址发送 跟action三种书写方式一致 (不写 写全称 写后缀)
    type:'get/post',  # 提交方式 默认get 跟form表单method参数一致
    data:{'uauername':'jason','password':123},  # 要发送的数据
    success:function(args(形参 写啥都可以){
        # 异步回调处理机制
    }
})
"""

当你在利用ajax进行前后端交互的时候
后端无论返回什么都只会被回调函数接收 而不再影响这个浏览器页面了


# 扩展 参数  代码发布项目还会涉及
dataType:'JSON'
"""
当后端是以HttpResponse返回的json格式的数据
默认是不会自动反序列化的
    1、自己手动JSON.parse()
    2、配置dataType参数
"""

# 结论:
    写ajax的时候 你可以直接将dataType参数加上,以防万一 
    或者后端就用JsonResonse
    
$.ajax({
    url:'',  # 朝后端哪个地址发送 跟action三种书写方式一致 (不写 写全称 写后缀)
    type:'get/post',  # 提交方式 默认get 跟form表单method参数一致
    dataType:'JSON',
    data:{'uauername':'jason','password':123},  # 要发送的数据
    success:function(args(形参 写啥都可以){
        # 异步回调处理机制
    }
})

 

小例子

"""
页面上有三个input框
    在前两个框中输入数字 点击按钮 朝后端发送ajax请求
    后端计算出结果 再返回给前端 动态展示到第三个input框中
    (整个过程页面不准有刷新 也不能在前端计算)
"""

<script>
    // 先给按钮绑定一个点击事件
    $('#btn').click(function () {
        // 朝后端发送ajax请求
        $.ajax({
            // 1、指定朝哪个后端发送ajax请求
            url:'',  // 不写就是朝当前地址提交
            // 2、请求方式
            type:'post',  // 不指定默认就是get 都是小写
            // 3、数据
            {#data:{'username':'jason','password':123},#}
            {#data:{'i1':$('#d1').val(),'i2':$('#d2').val()},#}
            // 4、回调函数(异步回调机制):当后端给你返回结果的时候会自动触发 args接收后端的返回结果
            success:function (args) {
                {#alert(args)  // 通过DOM操作动态渲染到第三个input框里面#}
                {#$('#d3').val(args)#}
                console.log(typeof args)
            }
        })
    })

</script>
                
                
"""
针对后如果是用HttpResponse返回的数据 回调函数不会自动帮你反序列化
如果后端直接用的是JsonResponse返回的数据 回调函数会自动帮你反序列化

HttpResponse解决方式
    1、自己在前端利用JSON.parse()
    2、在ajax里面配置一个参数
        后面再讲
"""

 

前后端传输数据的编码格式(contentType)

# 我们主要研究post请求数据的编码格式
"""
get请求数据就是直接放在url后面的
url?username=jason&password=123
"""

# 可以朝后端发送post请求的方式
    """
    1、form表单
    2、ajax请求
    """
    
    
"""
前后端传输数据的编码格式
    urlencode:默认的 ---> 从request.POST取提交的数据
    
    formdata:上传文件的  ---> 从request.POST取提交的数据
    
    json:ajax发送json格式数据  ---> request.POST取不出数据了
    
使用ajax和form表单 默认都是urlencoded格式

"""
# 研究form表单
    默认的数据编码格式是urlencoded
    数据格式:username=jason&password=123
    """
    django后端针对 符合 urlencoded编码格式的数据 都 会自动帮你解析封装到requet.POST中
    """
    
    如果把编码格式改为formdata,那么针对普通的键值对还是解析到request.POST中
    而将文件解析到request.FILES中
    
    
    form表单是没有办法发送json格式的数据的
    
    
   
# 研究ajax
    默认的编码格式也是urlencoded
    数据格式:username=jason&age=20
    """
    django后端针对 符合 urlencoded编码格式的数据 都 会自动帮你解析封装到requet.POST中
    """
  
    

 

ajax发送json格式数据

"""
前后端传输数据的时候 一定要确保编码格式跟数据真正的格式是一致的
不要骗人家!!!

{"username":"jason","age":25}   
    在request.POST里面肯定找不到
    
    django针对json
    
request对象方法补充
    request.is_ajax()
        判断当前请求是否是ajax请求 返回布尔值
"""

# 前端:
<script>
    $('#d1').click(function () {
        $.ajax({
            url:'',
            type:'post',
            data:JSON.stringify({'username':'jason','age':25}),
            contentType:'application/json',  // 指定编码格式
            success:function () {
            }
        })
    })
</script>

# 后端
# ajax发送json格式数据
import json
def ab_json(request):
    if request.is_ajax():
        print(request.POST)
        print(request.FILES)
        print(request.is_ajax())
        print(request.body)  # b'{"username":"jason","age":25}'
        # 针对json格式数据需要你自己手动处理
        json_bytes = request.body
        # json_str = json_bytes.decode('utf-8')
        # json_dict = json.loads(json_str)
        # print(json_dict,type(json_dict))  # {'username': 'jason', 'age': 25} <class 'dict'>

        # json.loads括号内如果传入了一个二进制格式的数据
        # 那么内部可以自动解码再反序列化
        json_dict = json.loads(json_bytes)
        print(json_dict,type(json_dict))
    return render(request,'ab_json.html')


"""
ajax发送json格式数据需要注意点
    1、contentType参数指定成:application/json
    2、数据是真正的json格式数据
    3、django后端不会帮你处理json格式数据需要你自己去requset.body获取并处理
"""

 

ajax发送文件

 

 

 

"""
ajax发送文件需要借助于js内置对象FormData
"""

# 前端
<body>
<p>username:<input type="text" id="d1"></p>
<p>password:<input type="password" id="d2"></p>
<p><input type="file" id="d3"></p>
<button class="btn btn-info" id="d4">点我</button>

<script>
    // 点击按钮朝后端发送普通键值对 和文件数据
    $('#d4').on('click',function () {
        // 1、需要先利用FormData内置对象
        let formDataObj = new FormData();
        // 2、添加普通的键值对
        formDataObj.append('username',$('#d1').val())
        formDataObj.append('password',$('#d2').val())
        // 3、添加文件对象
        formDataObj.append('myfile',$('#d3')[0].files[0])
        // 4、将对象基于ajax发送给后端
        $.ajax({
            url:'',
            type:'post',
            data:formDataObj,  //直接将对象放在data后面即可

            // ajax发送文件必须要指定的两个参数
            contentType:false,  // 不需要使用如何编码 django后端能够自动识别formdata对象
            processData:false,  // 告诉你的浏览器不要对你的数据进行任何处理

            success:function (args){

            }

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


# 后端
# ajax发送文件
def ab_file(request):
    if request.is_ajax():
        if request.method == 'POST':
            print(request.POST)
            print(request.FILES)
    return render(request,'ab_file.html')


"""
总结:
    1、需要利用内置对象FormData
        // 2、添加普通的键值对
        formDataObj.append('username',$('#d1').val())
        formDataObj.append('password',$('#d2').val())
        // 3、添加文件对象
        formDataObj.append('myfile',$('#d3')[0].files[0])
    2、需要指定两个关键性的参数
        contentType:false,  // 不需要使用如何编码 django后端能够自动识别formdata对象
        processData:false,  // 告诉你的浏览器不要对你的数据进行任何处理
    3、django后端能够直接识别到formdata对象并且能够
    将内部的普通键值自动解析并封装到request.POST中 
    将内部的文件数据自动解析并封装到request.FILES中
    
"""

 

 

django自带的序列化组件(drf做铺垫)

"""
如果发现你可以直接使用MySQL但是无法使用sqlite3
不要慌张不要恐惧 你只需要按照之前MySQL的操作将sqlite3的驱动装一下即可
"""

# 需求:在前端给我获取到后端用户表里面所有的数据 并且要是列表套字典


# 序列化组件相关
from app01 import models
import json
from django.http import JsonResponse
from django.core import serializers
def ab_ser(request):
    user_queryset = models.User.objects.all()
    # [{},{},{},{}]
    # user_list = []
    # for user_obj in user_queryset:
    #     tmp = {
    #         'pk':user_obj.pk,
    #         'username':user_obj.username,
    #         'age':user_obj.age,
    #         'gender':user_obj.get_gender_display()
    #     }
    #     user_list.append(tmp)
    # return JsonResponse(user_list,safe=False)
    # return render(request,'ab_ser.html',locals())
    """
    [{"pk": 1, "username": "jason", "age": "25", "gender": "male"}, 
    {"pk": 2, "username": "egon", "age": "31", "gender": "female"}, 
    {"pk": 3, "username": "keven", "age": "32", "gender": "others"}, 
    {"pk": 4, "username": "tank", "age": "40", "gender": 4}
    ]

    前后端分离的项目
        作为后端开发的你只需要写代码将数据处理好
        能够序列化返回给前端即可
            再写一个接口文档 告诉前端每个字段代表的意思即可
    """

    # 序列化
    res = serializers.serialize('json',user_queryset)
    """
    会自动帮你把数据变成json格式的字符串 并且内部非常的全面
    不好用  字段不能控制
    
    目前阶段 要做序列化  for循环拼列表套字典
    """
    return HttpResponse(res)
    """
    [{"model": "app01.user", 
    "pk": 1, 
    "fields": {"username": "jason", "age": "25", "gender": 1}}, 
    
    {"model": "app01.user", 
    "pk": 2, 
    "fields": {"username": "egon", "age": "31", "gender": 2}}, 
    
    {"model": "app01.user", 
    "pk": 3, 
    "fields": {"username": "keven", "age": "32", "gender": 3}}, 
    
    {"model": "app01.user", 
    "pk": 4,
     "fields": {"username": "tank", "age": "40", "gender": 4}}]
    """
"""
写接口 
    就是利用序列化组件渲染数据 然后写一个接口文档 该交代的交代一下就完事了
"""

 

ajax结合sweetalert实现删除按钮的二次确认操作

"""
sweetalert网址:
    https://lipis.github.io/bootstrap-sweetalert/
"""

"""
自己要学会如何拷贝
学会基于别人的基础之上做修改
研究各个参数表示的意思 然后照葫芦画瓢
"""

<script>
    $('.del').on('click',function () {
        // 先将当前标签对象存储起来
        let currentBtn = $(this);

        // 二次确认弹框
        swal({
              title: "你确定要删除吗",
              text: "你可要考虑清除哦,可能需要拎包跑路哦",
              type: "warning",
              showCancelButton: true,
              confirmButtonClass: "btn-danger",
              confirmButtonText: "是的,老子就要删!",
              cancelButtonText: "算了,算了!!!",
              closeOnConfirm: false,
              closeOnCancel: false,
              showLoaderOnConfirm: true
            },
            function(isConfirm) {
              if (isConfirm) {
                  // 朝后端发送ajax请求删除数据之后 再弹下面的提示框
                  $.ajax({
                      {#url:'/delete/user/' + currentBtn.attr('delete_id'),  // 传递主键值方式1#}
                      url:'/delete/user/',  //2 放在请求体里面
                      type: 'post',
                      data:{'delete_id':currentBtn.attr('delete_id')},
                      success:function (args) {  //args = {'code':'','msg':''}
                            //判断响应状态码 然后做不同的处理
                          if(args.code == 1000){
                              swal('删了',args.msg,'success');
                              // 1、lowb版本 直接刷新当前页面
                              {#window.location.reload();#}
                              // 2、利用DOM操作 动态刷新
                              currentBtn.parent().parent().remove()
                          }else{
                              swal('完了','出现了未知的错误','info');
                          }
                      }

                  })

                swal("删除了!", "赶紧跑路叭", "success");
              } else {
                swal("怂比", "不要说我认识你", "error");
              }
            });
    })
</script>

 

 

页面器

"""
django中有自带的分页器模块 但是书写起来很麻烦并且功能太简单
所以我们自己想发设发的写自定义分页器
https://www.cnblogs.com/Dominic-Ji/articles/12035722.html
"""

 

 

 

 

 

 

posted @ 2021-06-11 16:17  Palpitate~  阅读(17)  评论(0编辑  收藏  举报