您好,登錄后才能下訂單哦!
Python框架Django有著諸多優點,它提供的models可以讓開發者方便地操作數據庫,但正是由于對上層的良好的封裝,使得提升數據庫操作性能必須要清楚地知道Django的數據庫操作到底執行了哪些SQL語句。
例如數據更新操作,對單條記錄,可以使用save或者是update兩種方式
在Django工程下的settings.py下將log設置為DEBUG,即可查看save和update分別執行了哪些SQL語句
如有一張表名叫做Example
使用save:
k = Example.objects.get(id=481) k.total_calories = 12 k.save()
執行的SQL語句如下所示:
SELECT (1) AS `a` FROM `Example` WHERE `
Example`.`id` = 481 LIMIT 1; args=(481,)
UPDATE `Example` SET `user_id` = asdfasdf, `event_id` = -1, `join_type` = 0, `name` = , `phon
e` = , `email` = , `company_name` = , `address` = , `if_type` = 0, `code` = , `l
ocation` = , `total_days` = 0, `total_length` = 0, `total_calories` = 12, `comme
nts` = , `reserved_1` = , `reserved_2` = , `reserved_3` = , `reserved_4` = , `re
served_5` = , `create_datetime` = 2015-02-02 17:43:53 WHERE `Example`
.`id` = 481 ; args=(u'asdfasdf', -1, 0, u'', u'', u'
', u'', u'', 0, u'', u'', 0, 0, 12, u'', u'', u'', u'', u'', u'', u'2015-02-02 1
7:43:53', 481)
首先要查詢k這條記錄,然后save()的時候提交更新的內容,發現更新的時候把Example中的有字段都SET賦值的一次
使用update
Example.objects.filter(id=481).update(total_calories = 10)
執行的SQL語句是:
UPDATE `Example` SET `total_calories` = 1 0 WHERE (`Example`.`user_id` = asdfasdf AND `Example`.`id` = 481 ); args=(10, u'asdfasdf', 481)
這條SQL語句簡短而且執行速度要優于使用save的速度。
從SQL的執行情況來看,使用upate是要優于save方式的。
從使用情境上看,update更加適用于批量數據更新,而save則更適合當然也只適合做單條記錄的數據更新操作了。
在使用Django的數據模型操作數據庫時,了解這些底層的SQL操作很有必要。
補充知識:如何理解Django的save(commit=False)方法和save_m2m()方法
什么時候使用save(commit=False)方法,save_m2m方法以及如何使用是Django表單forms進階必需了解的知識。我們今天就帶你來看一看。
何時使用save(commit=False)方法
Stackoverflow上其實已經有了一段非常精煉的答案。英文原文如下,我把它翻譯了一下:
That's useful when you get most of your model data from a form, but need to populate some null=False fields with non-form data. Saving with commit=False gets you a model object, then you can add your extra data and save it.
當你通過表單獲取你的模型數據,但是需要給模型里null=False字段添加一些非表單的數據,該方法會非常有用。如果你指定commit=False,那么save方法不會理解將表單數據存儲到數據庫,而是給你返回一個當前對象。這時你可以添加表單以外的額外數據,再一起存儲。
save(commit=False)方法實際應用案例
下面我們來看一個實際應用案例。我們創建了一個叫文章Article的模型,里面包含title, body和作者author等多個字段,其中author字段非空null=False。我們由Article模型創建了一個ArticleForm表單,可以讓用戶發表新文章,但是我們故意把author字段除外了,因為我們不希望用戶編輯作者。
最后用戶提交的表單數據里肯定沒有author,當這樣的數據提交到數據庫時肯定會有問題的。所以我們先通過 article = form.save(commit=False)創建article實例,此時讓Django先不要發送數據到數據庫,等待我們把author添加好后,再把數據一起存儲到數據庫中。
下面是視圖文件views.py的代碼。最重要的是ArticleForm構成和article_create方法。
from .models import Article from django.forms import ModelForm from django.http import HttpResponseRedirect from django.shortcuts import render class ArticleForm(ModelForm): class Meta: model = Article exclude = ['author'] def article_create(request): if request.method == 'POST': form = ArticleForm(request.POST) if form.is_valid(): article = form.save(commit=False) # commit=False告訴Django先不提交到數據庫. article.author = request.user # 添加額外數據 article.save() # 發送到數據庫 return HttpResponseRedirect("/blog/") else: form = ArticleForm() return render(request, 'blog/article_create_form.html', {'form': form})
如果你使用Django自帶的基于類的視圖(CBV), 你可以使用form_valid方法完成上述同樣的操作。具體代碼如下。
from django.views.generic.edit import CreateView from .models import Article from django.forms import ModelForm # Create your views here. class ArticleForm(ModelForm): class Meta: model = Article exclude = ['author'] class ArticleCreateView(CreateView): model = Article form_class = ArticleForm template_name = 'blog/article_create_form.html' # Associate form.instance.user with self.request.user def form_valid(self, form): form.instance.author = self.request.user return super().form_valid(form)
何時使用save_m2m方法及如何使用
save_m2m方法只用來存儲多對多的關系。當你同時滿足下面兩個條件時,你需要使用此方法。如果你直接使用save()或form_valid()方法,是可以直接存儲多對多(m2m)關系的,不需要用save_m2m。
你使用了save(commit=False)方法
你的model里有多對多的關系(比如tags)
假設我們文章模型里有tags這個多對多的字段,我們的article_create方法需要增加一行。
def article_create(request): if request.method == 'POST': form = ArticleForm(request.POST) if form.is_valid(): article = form.save(commit=False) # commit=False tells Django that "Don't send this to database yet. article.author = request.user # Set the user object here article.save() # Now you can send it to DB form.save_m2m() return HttpResponseRedirect("/blog/") else: form = ArticleForm() return render(request, 'blog/article_create_form.html', {'form': form})
以上這篇Django數據庫操作之save與update的使用就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。