Django如何与Ajax交互

作者:大江狗 时间:2023-05-09 12:53:11 

目录
  • 前后端传输数据的编码格式

  • Ajax提交urlencoded格式数据

  • Ajax通过FormData上传文件

  • Ajax提交Json格式数据

  • Ajax发送POST请求时如何通过CSRF认证

  • Django Ajax案例1:联动下例菜单

  • Django Ajax案例2:Ajax上传文件

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

前后端传输数据的编码格式主要有三种, 本文接下来将详细演示。


urlencoded
formdata
json

Ajax提交urlencoded格式数据

Ajax给后台发送数据的默认编码格式是urlencoded,比如username=abcde&password=123456的形式。Django后端拿到符合urlencoded编码格式的数据都会自动帮你解析分装到request.POST中,与form表单提交的数据相同。

下面两种方式是等同的。


//手动构造数据data
$("#btnSubmit").click(function () {
   $.ajax({
       url: '/auth/', //也可以反向解析{% url 'login' %}
       type: 'post',
       data: {
           'username': $("#id_username").val(),
           'password': $('#id_password').val()
       },
       success: function (data){

}
   });
};

// .serialize() 方法可将<input>, <textarea> 以及 <select>表单序列化
// 成urlencoded格式数据

$("#btnSubmit").click(function () {
   let data = $("#loginForm").serialize();
   $.ajax({
       url: "/auth/", //别忘了加斜杠
       type: $("#loginForm").attr('method'),
       data: data,
       success: function (data) {

}
   });
});

Ajax通过FormData上传文件

Ajax上传文件需要借助于js内置对象FormData,另外上传文件时表单千万别忘了加enctype="multipart/form-data"属性。


//案例1,点击submi上传文件
$("#submitFile").click(function () {
   let formData = new FormData($("#upload-form"));
   $.ajax({
      url:"/upload/",//也可以写{% url 'upload' %}
      type:"post",
      data:formData,
      //这两个要必须写
      processData:false,  //不预处理数据  因为FormData 已经做了
      contentType:false,  //不指定编码了 因为FormData 已经做了
      success:function(data){
            console.log(data);
      }
   });
});

//案例2,同时上传文件并提交其它数据
$("#submitFile").click(function () {
   //js取到文件
   let myfile = $("#id_file")[0].files[0];
   //生成一个FormData对象
   let formdata = new FormData();
   //加值
   formdata.append('name', $("#id_name").val());
   //加文件
   formdata.append('myfile', myfile);
   $.ajax({
       url: '/upload/', //url别忘了加/杠
       type: 'post',
       //这两个要必须写
       processData:false,  //不预处理数据  因为FormData 已经做了
       contentType:false,  //不指定编码了 因为FormData 已经做了
       data: formdata,
       success: function (data) {
           console.log(data);
       }
   });
});

Ajax提交Json格式数据

前后端传输数据的时候一定要确保声明的编码格式跟数据真正的格式是一致的。如果你通过Ajax发送Json格式数据给Django后端,请一定注意以下三点:

  1. contentType参数指定成application/json;

  2. 数据是真正的json格式数据;

  3. Django后端不会帮你处理json格式数据需要你自己去request.body获取并处理。


$("#submitBtn").click(function () {
   var data_obj={'name':'abcdef','password':123456};//Javascript对象
   $.ajax({
       url:'',
       type:'post',
       contentType:'application/json',  //一定要指定格式 contentType
       data:JSON.stringify(data_obj),    //转换成json字符串格式
       success:function (data) {
           console.log(data)
       }
   });
});

Ajax发送POST请求时如何通过CSRF认证


// 第一种方式直接在发送数据中加入csrfmiddlewaretoken
$("#btn").on("click",function () {
   $.ajax({
       url:"/some_url/",
       type:"POST",
       data:{
           csrfmiddlewaretoken: {{ csrf_token }}, //写在模板中,才会被渲染
       },
       success:function (data) {
   }
});
});

//通过jquery选择器获取csrfmiddlewaretoken
$("#btn").on("click",function () {
   $.ajax({
       url:"/some_url/",
       type:"POST",
       data:{
           csrfmiddlewaretoken:$('[name="csrfmiddlewaretoken"]').val(),
       },
       success:function (data) {

}
   });
});

//使用jquery.cookie.js调用请求头cookie中的csrftoken
<script src="/static/jquery.cookie.js">
<script>
    $("#btn").on("click",function () {
    $.ajax({
       url:"/some_url/",
       type:"POST",
       headers:{"X-CSRFToken":$.cookie('csrftoken')},
       data:$("#form1").serialize()
   });
  })
</script>

Django Ajax案例1:联动下例菜单

联动下拉菜单是Web开发中一个被经常使用的应用。比如当你从一个列表从选择一个国家的时候,联动下拉菜单会同步显示属于该国家所有城市列表供用户选择。今天我们就教你如何使用Django+Ajax生成联动下拉菜单。

我们的模型如下所示:


class Country(models.Model):
   name = models.CharField(verbose_name="国家", max_length=50)

   def __str__(self):
       return self.name

class City(models.Model):
   name = models.CharField(verbose_name="城市", max_length=50)
   country = models.ForeignKey(Country, on_delete=models.CASCADE, verbose_name="国家",)

   def __str__(self):
       return self.name

我们的模板如下所示,表单中对应国家和城市下拉菜单的DOM元素id分别为id_country和id_city。当用户选择国家后,ajax会携带国家的id值向后台发送请求获得当前国家的所有城市清单,并在前端渲染显示。


{% block content %}
<h2>创建用户 - 联动下拉菜单</h2>
<form method="post" class="form-horizontal" role='form' action="">
 {% csrf_token %}
 {{ form.as_p }}
 <button type="submit" class="btn btn-primary">Submit</button>
</form>
{% endblock %}

<script src="https://code.jquery.com/jquery-3.1.0.min.js"></script>
<script>
   $("#id_country").change(function() {
     var country_id = $(this).val();

     $.ajax({
       url: '/ajax/load_cities/',
       data: {
         'country_id': country_id
       },
       type: 'GET',
       dataType: 'json',
       success: function (data) {
           var content='';
           $.each(data, function(i, item){
                 content+='<option value='+item.id+'>'+item.name+'</option>'
               });
           $('#id_city').html(content)
       },

     });
   });
 </script>

Django负责处理视图Ajax请求的视图函数如下所示:


def ajax_load_cities(request):
   if request.method == 'GET':
       country_id = request.GET.get('country_id', None)
       if country_id:
           data = list(City.objects.filter(country_id=country_id).values("id", "name"))
           return JsonResponse(data, safe=False)

Django Ajax案例2:Ajax上传文件

前端模板及js文件如下所示, 请注意我们是如何在表单中加入了enctype属性,如何使用FormData上传文件,并解决了csrftoken问题的。


{% block content %}
<form action="" method="post" enctype="multipart/form-data" id="form">
   <ul class="errorlist"></ul>
   {% csrf_token %}
{{ form.as_p }}
<input type="button" class="btn btn-info form-control" value="submit" id="btn" />
</form>
<table class="table table-striped" id="result">
</table>
{% endblock %}

{% block js %}
<script src=" https://cdn.jsdelivr.net/jquery.cookie/1.4.1/jquery.cookie.min.js ">
</script>
<script>
var csrftoken = $.cookie('csrftoken');
function csrfSafeMethod(method) {
   return (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));
}
$(document).ready(function(){
  $('#btn').click(function(e){
       e.preventDefault();
       // 构建FormData对象
       var form_data = new FormData();
       form_data.append('file', $('#id_file')[0].files[0]);
       $.ajax({
       url: '/file/ajax_upload/',
       data: form_data,
       type: 'POST',
       dataType: 'json',
       // 告诉jQuery不要去处理发送的数据, 发送对象。
       processData : false,
       // 告诉jQuery不要去设置Content-Type请求头
       contentType : false,
       // 获取POST所需的csrftoken
       beforeSend: function(xhr, settings) {
           if (!csrfSafeMethod(settings.type) && !this.crossDomain) {
           xhr.setRequestHeader("X-CSRFToken", csrftoken);
           }},
       success: function (data) {
           if(data['error_msg']) {
               var content = '<li>'+ data['error_msg'] + '</li>';
               $('ul.errorlist').html(content);
           }
           else
           {
           var content= '<thead><tr>' +
           '<th>Name and URL</th>' +
           '<th>Size</th>' +
           '</tr></thead><tbody>';
            $.each(data, function(i, item) {
                 content = content +
                 '<tr><td>' +
                 "<a href= ' " +
                 item['url'] +
                 " '> " +
                 item['url'] +
                 '</a></td><td>' +
                 item['size'] +
                 '</td><td><tr>'
               });
            content = content + "</tbody>";
            $('#result').html(content);
            }
          },
       });
  });
});
 </script>
{% endblock %}

Django负责处理视图Ajax请求的视图函数如下所示:


# handling AJAX requests
def ajax_upload(request):
   if request.method == "POST":
       form = FileUploadModelForm(data=request.POST, files=request.FILES)
       if form.is_valid():
           form.save()
           # Obtain the latest file list
           files = File.objects.all().order_by('-id')
           data = []
           for file in files:
               data.append({
                   "url": file.file.url,
                   "size": filesizeformat(file.file.size),
                   })
           return JsonResponse(data, safe=False)
       else:
           data = {'error_msg': "Only jpg, pdf and xlsx files are allowed."}
           return JsonResponse(data)
   return JsonResponse({'error_msg': 'only POST method accpeted.'})

来源:https://pythondjango.cn/django/basics/16-django-ajax/

标签:Django,Ajax
0
投稿

猜你喜欢

  • ORACLE应用经验(2)

    2010-07-31 13:31:00
  • Python基于opencv的简单图像轮廓形状识别(全网最简单最少代码)

    2023-06-23 06:59:14
  • Pandas中describe()函数的具体使用

    2021-03-22 14:32:12
  • 浅谈MySQL数据库崩溃(crash)的常见原因和解决办法

    2024-01-17 12:52:58
  • Python 常用 PEP8 编码规范详解

    2022-09-03 06:21:12
  • Dreamweaver使用技巧--让css使网页图片半透明

    2010-09-05 21:13:00
  • Mootools 1.2教程(16)——排序类和方法简介

    2008-12-10 14:18:00
  • Git远程仓库配置SSH的实现(以github为例)

    2023-10-21 18:16:00
  • 如何使用PHP中的字符串函数

    2024-05-11 10:01:52
  • python 3.3 下载固定链接文件并保存的方法

    2023-09-22 16:43:50
  • 基于python+selenium自动健康打卡的实现代码

    2022-04-28 06:31:28
  • asp.net实现图片以二进制流输出的两种方法

    2023-06-28 21:49:02
  • 在sql中实现取一行最大值或者最小值

    2024-01-24 21:34:28
  • Pandas实现DataFrame按行求百分数(比例数)

    2022-10-23 10:49:01
  • python lxml中etree的简单应用

    2022-01-11 09:34:15
  • Python群发邮件实例代码

    2021-05-05 18:42:35
  • MySQL一个语句查出各种整形占用字节数及最大最小值的实例

    2024-01-25 11:30:43
  • Laravel框架实现点播上传阿里云功能

    2023-06-13 20:13:30
  • Java与Mysql锁相关知识总结

    2024-01-17 14:24:19
  • python的函数最详解

    2022-02-15 17:05:05
  • asp之家 网络编程 m.aspxhome.com