Django中的Model操作表的實(shí)現(xiàn)
Model 操作表
一、基本操作
# 增
models.Tb1.objects.create(c1='xx', c2='oo') #增加一條數(shù)據(jù),可以接受字典類(lèi)型數(shù)據(jù) **kwargs
obj = models.Tb1(c1='xx', c2='oo')
obj.save()
dic = {'c1':'xx','c2':'oo'}
models.Tb1.objects.create(**dic) #Form的產(chǎn)出結(jié)果是一個(gè)字典,可以根據(jù)這個(gè)Form的字典和**直接在數(shù)據(jù)庫(kù)創(chuàng)建數(shù)據(jù)
# 查
models.Tb1.objects.get(id=123) # 獲取單條數(shù)據(jù),不存在則報(bào)錯(cuò)(不建議)
models.Tb1.objects.all() # 獲取全部 .first() 取第一條數(shù)據(jù)
models.Tb1.objects.filter(name='seven') # 獲取指定條件的數(shù)據(jù) 也可以用**的方式傳參數(shù)
# 刪
models.Tb1.objects.filter(name='seven').delete() # 刪除指定條件的數(shù)據(jù)
# 改
models.Tb1.objects.filter(name='seven').update(gender='0') # 將指定條件的數(shù)據(jù)更新,均支持 **kwargs
obj = models.Tb1.objects.get(id=1)
obj.c1 = '111'
obj.save() # 修改單條數(shù)據(jù)
細(xì)看從數(shù)據(jù)庫(kù)取出的數(shù)據(jù)類(lèi)型 :
w = models.Simp.objects.all() print w, type(w) [<Simp: chenc>, <Simp: zan>, <Simp: zhangsan>] <class 'django.db.models.query.QuerySet'>
可以看到,從數(shù)據(jù)庫(kù)取出個(gè)數(shù)據(jù)看起來(lái)像包含對(duì)象的列表。而實(shí)際上整個(gè)數(shù)據(jù)為django中的特殊類(lèi)型QuerySet。
.all()是取得所有列的數(shù)據(jù),可以加.values()取出某一列,每一項(xiàng)的值為一個(gè)字典:
w = models.Simp.objects.all().values('username')
print w, type(w)
[{'username': u'chenc'}, {'username': u'zan'}, {'username': u'zhangsan'}] <class 'django.db.models.query.QuerySet'>
.values_list(),獲取到的值為一個(gè)元組
w = models.Simp.objects.all().values_list('username')
print w, type(w)
[(u'chenc',), (u'zan',), (u'zhangsan',)] <class 'django.db.models.query.QuerySet'>
.values_list()也可以添加多個(gè)參數(shù):(可以配合Form在前端生成動(dòng)態(tài)的select)
w = models.Simp.objects.all().values_list('id', 'username')
print w, type(w)
[(1, u'chenc'), (2, u'zan'), (3, u'zhangsan')] <class 'django.db.models.query.QuerySet'>
query可以查看執(zhí)行的sql語(yǔ)句:
b = models.Simp.objects.all() print b.query SELECT "app01_simp"."id", "app01_simp"."username", "app01_simp"."password" FROM "app01_simp"
二、進(jìn)階操作
利用雙下劃線將字段和對(duì)應(yīng)的操作連接起來(lái)
# 獲取個(gè)數(shù)
#
# models.Tb1.objects.filter(name='seven').count()
# 大于,小于
#
# models.Tb1.objects.filter(id__gt=1) # 獲取id大于1的值
# models.Tb1.objects.filter(id__lt=10) # 獲取id小于10的值
# models.Tb1.objects.filter(id__lt=10, id__gt=1) # 獲取id大于1 且 小于10的值
# in
#
# models.Tb1.objects.filter(id__in=[11, 22, 33]) # 獲取id等于11、22、33的數(shù)據(jù)
# models.Tb1.objects.exclude(id__in=[11, 22, 33]) # not in
# contains
#
# models.Tb1.objects.filter(name__contains="ven")
# models.Tb1.objects.filter(name__icontains="ven") # icontains大小寫(xiě)不敏感
# models.Tb1.objects.exclude(name__icontains="ven")
# range
#
# models.Tb1.objects.filter(id__range=[1, 2]) # 范圍bettwen and
# 其他類(lèi)似
#
# startswith,istartswith, endswith, iendswith,
# order by
#
# models.Tb1.objects.filter(name='seven').order_by('id') # asc 從小到大
# models.Tb1.objects.filter(name='seven').order_by('-id') # desc 從大到小
# limit 、offset
#
# models.Tb1.objects.all()[10:20]
# group by
from django.db.models import Count, Min, Max, Sum
# models.Tb1.objects.filter(c1=1).values('id').annotate(c=Count('num'))
# SELECT "app01_tb1"."id", COUNT("app01_tb1"."num") AS "c" FROM "app01_tb1" WHERE "app01_tb1"."c1" = 1 GROUP BY "app01_tb1"."id"
三、操作表之一對(duì)多實(shí)例
我們來(lái)創(chuàng)建一個(gè)完整的過(guò)程 特別聲明:(過(guò)程繁瑣無(wú)聊,純屬作者梳理思路,可以直接跳過(guò)看解析)
首先我們先來(lái)創(chuàng)造兩張表,并設(shè)置外鍵。
class GroupNew(models.Model): groupname = models.CharField(max_length=15) class UserNew(models.Model): user = models.CharField(max_length=15) 10 group = models.ForeignKey(GroupNew)
然后再組表里創(chuàng)建幾條數(shù)據(jù),配置好url生成數(shù)據(jù):
# coding:utf-8
from django.shortcuts import HttpResponse,render
from app01 import models
def Foreign(request):
models.GroupNew.objects.create(groupname='CEO')
models.GroupNew.objects.create(groupname='CTO')
models.GroupNew.objects.create(groupname='COO')
return HttpResponse('OK')
創(chuàng)建用于生成表單的form:(動(dòng)態(tài)的select)
# coding:utf-8
from django import forms
from app01 import models
class CreateForm(forms.Form):
user = forms.CharField(max_length=20)
group = forms.IntegerField(widget=forms.Select())
def __init__(self, *args, **kwargs):
super(CreateForm, self).__init__(*args, **kwargs)
self.fields['group'].widget.choices = models.GroupNew.objects.all().values_list('id', 'groupname')
在views.py里寫(xiě)好創(chuàng)建用戶的方法,并應(yīng)用form:
from app01.forms import foreign as UserForm
def createuser(request):
obj = UserForm.CreateForm()
return render(request, 'foreign/createuser.html', {'obj': obj})
創(chuàng)建一個(gè)html頁(yè)面,用來(lái)在前端操作創(chuàng)建用戶:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
</head>
<body>
<form action="/createuser/" method="post">
<p>{{ obj.user }}</p>
<p>{{ obj.group }}</p>
<input type="submit" value="提交" />
</form>
</body>
</html>
配置好url訪問(wèn)成功:

現(xiàn)在我們要來(lái)創(chuàng)建用戶數(shù)據(jù):(兩種方法)
def createuser(request):
obj = UserForm.CreateForm(request.POST)
if request.method == 'POST':
if obj.is_valid():
all_data = obj.clean()
group_id = all_data['group']
username = all_data['user']
# 1、先獲取字典表里的數(shù)據(jù)對(duì)象
# group_obj = models.GroupNew.objects.get(id=group_id)
# # 創(chuàng)建用戶數(shù)據(jù)
# models.UserNew.objects.create(user=username, group=group_obj)
# 2、 直接傳group_id不需獲取對(duì)象
models.UserNew.objects.create(user=username, group_id=group_id)
else:
pass
return render(request, 'foreign/createuser.html', {'obj': obj})
還有一種方法是根據(jù)Form的修改來(lái)創(chuàng)建:(注意:前端頁(yè)面也需要加上_id與之匹配{{ obj.group_id }} )
forms.py
# coding:utf-8
from django import forms
from app01 import models
class CreateForm(forms.Form):
user = forms.CharField(max_length=20)
group_id = forms.IntegerField(widget=forms.Select()) #根據(jù)數(shù)據(jù)庫(kù)存的字段來(lái)修改這里的form使之與數(shù)據(jù)庫(kù)相同
def __init__(self, *args, **kwargs):
super(CreateForm, self).__init__(*args, **kwargs)
self.fields['group_id'].widget.choices = models.GroupNew.objects.all().values_list('id', 'groupname')
然后再vires.py里面就可以直接拿到前端傳來(lái)的字典,直接創(chuàng)建數(shù)據(jù)了。注意用**的方式傳參
from app01.forms import foreign as UserForm
def createuser(request):
obj = UserForm.CreateForm(request.POST)
if request.method == 'POST':
if obj.is_valid():
all_data = obj.clean()
print all_data
# group_id = all_data['group']
username = all_data['user']
# 1、先獲取字典表里的數(shù)據(jù)對(duì)象
# group_obj = models.GroupNew.objects.get(id=group_id)
# # 創(chuàng)建用戶數(shù)據(jù)
# models.UserNew.objects.create(user=username, group=group_obj)
# 2、 直接傳group_id不需獲取對(duì)象
models.UserNew.objects.create(**all_data) #用**的方式傳遞字典參數(shù)
else:
pass
return render(request, 'foreign/createuser.html', {'obj': obj})
下面我們可以來(lái)展示一下數(shù)據(jù):
在views.pu里把數(shù)據(jù)全部拿到,返回到前端:
from app01.forms import foreign as UserForm
def createuser(request):
obj = UserForm.CreateForm(request.POST)
if request.method == 'POST':
if obj.is_valid():
all_data = obj.clean()
models.UserNew.objects.create(**all_data)
else:
pass
user_list = models.UserNew.objects.all() #獲取到表中的全部數(shù)據(jù)
return render(request, 'foreign/createuser.html', {'obj': obj, 'user_list': user_list})
四、前端和后臺(tái)獲取數(shù)據(jù) (了不起的雙下劃線“__”)
前端頁(yè)面獲取數(shù)據(jù)展示:
在前端頁(yè)面不能顯示用戶組的id,而是直接顯示用戶組的名稱(chēng)
因?yàn)橥怄I對(duì)應(yīng)的字段是一個(gè)對(duì)象,這個(gè)對(duì)象代指的是字典表中的一行數(shù)據(jù),所以我們?cè)谇岸巳〉阶值浔碇械臄?shù)據(jù)用“.”獲取
<table>
{% for item in user_list %}
<tr>
<td>{{ item.user }}</td>
<td>{{ item.group.groupname }}</td> 從對(duì)象中獲取數(shù)據(jù)用"."
</tr>
{% endfor %}
</table>
后臺(tái)獲取數(shù)據(jù):(了不起的雙下劃線)
我們?cè)谇岸擞胓et的方法獲取數(shù)據(jù),
在瀏覽器的url后追加字符 http://127.0.0.1:8000/createuser/?group=CEO
例如:我們要展示CEO組里面所以得用戶,在views里要如何寫(xiě)呢?
from app01.forms import foreign as UserForm
def createuser(request):
obj = UserForm.CreateForm(request.POST)
if request.method == 'POST':
if obj.is_valid():
all_data = obj.clean()
models.UserNew.objects.create(**all_data)
else:
pass
val = request.GET.get('group')
user_list = models.UserNew.objects.filter(group__groupname=val) # 去字典表里的數(shù)據(jù)時(shí)用雙下劃線
return render(request, 'foreign/createuser.html', {'obj': obj, 'user_list': user_list})
五、解析與總結(jié)
所有的聯(lián)表操作只需理解一點(diǎn):
model中一般字段為字符串,而外鍵代表的是一個(gè)對(duì)象,這個(gè)對(duì)象就是字典表中的一行數(shù)據(jù)。

創(chuàng)建數(shù)據(jù):
1、根據(jù)對(duì)象級(jí)別來(lái)操作表。
表中有外鍵的字段代指的就是另外一張表的一行數(shù)據(jù)。在Models里的表現(xiàn)形式就是一個(gè)對(duì)象,那么我們創(chuàng)建這張表時(shí),給外鍵的字段傳值時(shí)也應(yīng)該傳一個(gè)對(duì)象,而從前臺(tái)獲取到的是一個(gè)數(shù)字“1”,1不是一個(gè)對(duì)象,根據(jù)這個(gè)“1”先創(chuàng)建一個(gè)字典表的對(duì)象,所以根據(jù)前臺(tái)的“1”先獲取字典表里對(duì)應(yīng)的數(shù)據(jù)對(duì)象,把這個(gè)對(duì)象傳遞到創(chuàng)建表時(shí)所需的對(duì)象參數(shù)即可。
(其實(shí)也是把對(duì)象轉(zhuǎn)換成sql語(yǔ)句拼接成帶_id的樣式,有一種多此一舉的感覺(jué))
2、根據(jù)數(shù)據(jù)庫(kù)級(jí)別來(lái)操作表。
首先我們可以先查看一下數(shù)據(jù)庫(kù)表中的外鍵值的樣式 user_group_id?!癬id”在models沒(méi)有定義的,是django自動(dòng)為我們定義的樣式。所以我們?cè)谙驍?shù)據(jù)庫(kù)中插入數(shù)據(jù)的時(shí)候,外鍵的字段應(yīng)該加上_id,這樣就可以直接傳從前臺(tái)獲取的數(shù)字了。
(可以再form中修改外鍵的值,加"_id",這樣傳到后臺(tái)的數(shù)據(jù)就能直接拿到一個(gè)字典,在用**的方式傳值創(chuàng)建數(shù)據(jù))
獲取數(shù)據(jù):
跨表取數(shù)據(jù)用 "."
因?yàn)樵趍odels里的外鍵代指的是另外一張表的一行數(shù)據(jù),且這個(gè)數(shù)據(jù)類(lèi)型為一個(gè)對(duì)象。
那么就可以根據(jù)這個(gè)對(duì)象直接找到那行數(shù)據(jù)里對(duì)應(yīng)的字段。用 “.”獲取
在前端頁(yè)面就可以{{item.group.groupname}}方式獲取到。
跨表過(guò)濾數(shù)據(jù)時(shí)用 “__”
過(guò)濾數(shù)據(jù)用雙下劃線 “__”(group__groupname=val),如果有多張表建立外鍵繼續(xù)用雙下劃線“__”跨表查詢數(shù)據(jù)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python+appium自動(dòng)化測(cè)試之如何控制App的啟動(dòng)和退出
本文主要介紹了python+appium自動(dòng)化測(cè)試之如何控制App的啟動(dòng)和退出,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02
Python利用pywin32庫(kù)實(shí)現(xiàn)將PPT導(dǎo)出為高清圖片
這篇文章主要為大家詳細(xì)介紹了Python如何利用pywin32庫(kù)實(shí)現(xiàn)將PPT導(dǎo)出為高清圖片的功能,文中的示例代講解詳細(xì),感興趣的小伙伴可以了解一下2023-01-01
Python中往列表中插入字典時(shí),數(shù)據(jù)重復(fù)問(wèn)題
這篇文章主要介紹了Python中往列表中插入字典時(shí),數(shù)據(jù)重復(fù)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02
python 將大文件切分為多個(gè)小文件的實(shí)例
今天小編就為大家分享一篇python 將大文件切分為多個(gè)小文件的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-01-01
python實(shí)現(xiàn)判斷數(shù)組是否包含指定元素的方法
這篇文章主要介紹了python實(shí)現(xiàn)判斷數(shù)組是否包含指定元素的方法,涉及Python中in的使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07

