[轉]django自定義表單提交

来源:http://www.cnblogs.com/CQ-LQJ/archive/2016/01/15/5133671.html
-Advertisement-
Play Games

原文網址:http://www.cnblogs.com/retop/p/4677148.html註:本人使用的Django1.8.3版本進行測試除了使用Django內置表單,有時往往我們需要自定義表單。對於自定義表單Post方式提交往往會帶來由CSRF(跨站請求偽造)產生的錯誤"CSRF verif...


原文網址:http://www.cnblogs.com/retop/p/4677148.html

註:本人使用的Django1.8.3版本進行測試

除了使用Django內置表單,有時往往我們需要自定義表單。對於自定義表單Post方式提交往往會帶來由CSRF(跨站請求偽造)產生的錯誤"CSRF verification failed. Request aborted."

本篇博客只要針對"表單提交"和"Ajax提交"兩種方式來解決CSRF帶來的錯誤

一、表單提交

Template:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 <!DOCTYPE html> <html lang="en"> <head>     <meta charset="UTF-8">     <title>計算數字和</title> </head> <body>     <form method="post" action="{%url 'Calculate' %}">         {% csrf_token %}         <label for="A"><input id="A" name="ValueA" type="text"></label>         <label for="B"><input id="B" name="ValueB" type="text"></label>         <input type="submit" value="開始計算">     </form> </body> </html>

Views.py:

?
1 2 3 4 5 6 7 8 def Calculate(request):     if request.POST:         a=request.POST["ValueA"]         b=request.POST["ValueB"]         c=str(int(a)+int(b))         return  render_to_response('Result.html',{'result':c})     else:         return render_to_response('Calculation.html',context_instance=RequestContext(request))

需要註意:

  (1)在<form>標簽內添加{% csrf_token %},這樣在表單提交的過程中,會產生"csrfmiddlewaretoken"標識去防止CSRF

  (2)在Get請求頁面時,需要添加context_instance=RequestContext(request) ,它和{% csrf_token %}配合使用,缺少一個都會出現上述錯誤,RequestContext 需要在 django.shortcuts 導入

  (3)只有當表單以Post方式提交時,才需要驗證CSRF,Get方式是不需要的

二、Ajax提交

同比與表單提交,Ajax提交需要進行額外的操作,Ajax提交時需要自己提供"csrfmiddlewaretoken"標識參數。我們除了需要引入JQuery外還需要引入一段JS代碼

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 jQuery(document).ajaxSend(function(event, xhr, settings) {     function getCookie(name) {         var cookieValue = null;         if (document.cookie && document.cookie != '') {             var cookies = document.cookie.split(';');             for (var i = 0; i < cookies.length; i++) {                 var cookie = jQuery.trim(cookies[i]);                 // Does this cookie string begin with the name we want?                 if (cookie.substring(0, name.length + 1) == (name + '=')) {                     cookieValue = decodeURIComponent(cookie.substring(name.length + 1));                     break;                 }             }         }         return cookieValue;     }     function sameOrigin(url) {         // url could be relative or scheme relative or absolute         var host = document.location.host; // host + port         var protocol = document.location.protocol;         var sr_origin = '//' + host;         var origin = protocol + sr_origin;         // Allow absolute or scheme relative URLs to same origin         return (url == origin || url.slice(0, origin.length + 1) == origin + '/') ||             (url == sr_origin || url.slice(0, sr_origin.length + 1) == sr_origin + '/') ||             // or any other URL that isn't scheme relative or absolute i.e relative.             !(/^(\/\/|http:|https:).*/.test(url));     }     function safeMethod(method) {         return (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));     }         if (!safeMethod(settings.type) && sameOrigin(settings.url)) {         xhr.setRequestHeader("X-CSRFToken", getCookie('csrftoken'));     } });

Template:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 <!DOCTYPE html> <html lang="en"> <head>     <meta charset="UTF-8">     <title>Ajax 提交</title>     <script type="text/javascript" src="/static/jquery.js"></script>     <script type="text/javascript">         jQuery(document).ajaxSend(function(event, xhr, settings) {     function getCookie(name) {         var cookieValue = null;         if (document.cookie && document.cookie != '') {             var cookies = document.cookie.split(';');             for (var i = 0; i < cookies.length; i++) {                 var cookie = jQuery.trim(cookies[i]);                 // Does this cookie string begin with the name we want?                 if (cookie.substring(0, name.length + 1) == (name + '=')) {                     cookieValue = decodeURIComponent(cookie.substring(name.length + 1));                     break;                 }             }         }         return cookieValue;     }     function sameOrigin(url) {         // url could be relative or scheme relative or absolute         var host = document.location.host; // host + port         var protocol = document.location.protocol;         var sr_origin = '//' + host;         var origin = protocol + sr_origin;         // Allow absolute or scheme relative URLs to same origin         return (url == origin || url.slice(0, origin.length + 1) == origin + '/') ||             (url == sr_origin || url.slice(0, sr_origin.length + 1) == sr_origin + '/') ||             // or any other URL that isn't scheme relative or absolute i.e relative.             !(/^(\/\/|http:|https:).*/.test(url));     }     function safeMethod(method) {         return (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));     }        if (!safeMethod(settings.type) && sameOrigin(settings.url)) {         xhr.setRequestHeader("X-CSRFToken", getCookie('csrftoken'));     } });     </script>     <script type="text/javascript">         $(function(){              $.ajaxSetup({                     data:{csrfmiddlewaretoken: '{{ csrf_token }}'}                 });                 $("#Comment").click(function(){                     $.post('{% url 'AjaxRequest' %}',{"a":$("#A").val(),"b":$("#B").val()},function(data){                        $("#result").html(data);                     });                 });         });     </script> </head> <body>     <label for="A"><input id="A" name="ValueA" type="text"></label>     <label for="B"><input id="B" name="ValueB" type="text"></label>     <input type="button" id="Comment" value="開始計算">     <h1>計算的結果為:<span id="result"></span></h1> </body> </html>

View.py:

?
1 2 3 4 5 6 7 8 def AjaxRequest(request):     if request.POST:         a =request.POST["a"]         b=request.POST["b"]         c=int(a)+int(b)         return JsonResponse(c,safe=False)     else:         return render_to_response('AjaxDemo.html',context_instance=RequestContext(request))

需要註意:

  (1)在使用引入的JS代碼後,需要添加如下代碼,這樣JS就可以自動幫我們生成"csrfmiddlewaretoken"標識,接下來你就可以使用$.post()了

?
1 2 3 $.ajaxSetup({                    data:{csrfmiddlewaretoken: '{{ csrf_token }}'}                });

  (2)context_instance=RequestContext(request) 並不是必須的

  (3)Get請求不需要以上操作,直接使用$.get()即可

 

總結:本人學習Django的時間不長,寫博客的目的只要是為了自己做知識記錄和對知識的分享,如果哪裡寫的不好,還請廣大博友指點,多多包涵。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 第一種:DOM。DOM的全稱是Document Object Model,也即文檔對象模型。在應用程式中,基於DOM的XML分析器將一個XML文檔轉換成一個對象模型的集合(通常稱DOM樹),應用程式正是通過 對這個對象模型的操作,來實現對XML文檔數據的操作。通過DOM介面,應用程式可以在任何時候訪...
  • // TMemoryStream 轉化為string字元串function MemoryStreamToString(M: TMemoryStream): AnsiString;begin SetString(Result, PChar(M.Memory), M.Size div SizeOf(C....
  • 1、常見的兩種緩存本地緩存:不需要序列化,速度快,緩存的數量與大小受限於本機記憶體分散式緩存:需要序列化,速度相較於本地緩存較慢,但是理論上緩存的數量與大小無限(因為緩存機器可以不斷擴展)2、本地緩存Google guava cache:當下最好用的本地緩存Ehcache:spring預設集成的一個緩...
  • 常用方式: int a = 12; //註意:通常情況下,這個會設置成一個類變數,比如說Segement中的段鎖與copyOnWriteArrayList中的全局鎖 final ReentrantLock lock = new ReentrantLock()...
  • 最常用的方式: int a = 12; //註意:通常情況下,這個會設置成一個類變數,比如說Segement中的段鎖與copyOnWriteArrayList中的全局鎖 final ReentrantLock lock = new ReentrantLock...
  • 初次接觸python,寫的很簡單,開發工具PyCharm,python 3.4很方便python 部分模塊安裝時需要其他的附屬模塊之類的,可以先pip install wheel然後可以直接下載whl文件進行安裝pip installlxml-3.5.0-cp34-none-win32.whl定義一...
  • 在多線程應用中鎖是一個很簡單又很複雜的技術,之所以要用到鎖是因為在多進程/線程環境下,一段代碼可能會被同時訪問到,如果這段代碼涉及到了共用資源(數據)就需要保證數據的正確性。也就是所謂的線程安全。之前寫過一篇著於Java線程安全的博客:鏈接我是在寫一個服務端程式時應用到讀寫鎖,在一個記憶體緩存。先來看...
  • print ('hello world!')1. 下載python,並設置path系統環境變數;當在命令行中輸入python,出現如下界面,顯示安裝成功。2. 安裝最新的sublime,使用註冊機破解,或者Help->中輸入license,提供三個親測可用的激活碼,在文章最後。 打開sublime....
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...