文章出處 https://www.cnblogs.com/wupeiqi/articles/5246483.html
Model
到目前為止,當我們的程式涉及到資料庫相關操作時,我們一般都會這麼搞:
- 建立資料庫,設計表結構和字段
- 使用 MySQLdb 來連接配接資料庫,并編寫資料通路層代碼
- 業務邏輯層去調用資料通路層執行資料庫操作
import MySQLdb
def GetList(sql):
db = MySQLdb.connect(user='root', db='wupeiqidb', passwd='1234', host='localhost')
cursor = db.cursor()
cursor.execute(sql)
data = cursor.fetchall()
db.close()
return data
def GetSingle(sql):
db = MySQLdb.connect(user='root', db='wupeiqidb', passwd='1234', host='localhost')
cursor = db.cursor()
cursor.execute(sql)
data = cursor.fetchone()
db.close()
return data
View Code
django為使用一種新的方式,即:關系對象映射(Object Relational Mapping,簡稱ORM)。
PHP:activerecord
Java:Hibernate
C#:Entity Framework
django中遵循 Code Frist 的原則,即:根據代碼中定義的類來自動生成資料庫表。
一、建立表
1、基本結構
?
1 2 3 4 5 6 | |
AutoField(Field)
- int自增列,必須填入參數 primary_key=True
BigAutoField(AutoField)
- bigint自增列,必須填入參數 primary_key=True
注:當model中如果沒有自增列,則自動會建立一個列名為id的列
from django.db import models
class UserInfo(models.Model):
# 自動建立一個列名為id的且為自增的整數列
username = models.CharField(max_length=32)
class Group(models.Model):
# 自定義自增列
nid = models.AutoField(primary_key=True)
name = models.CharField(max_length=32)
SmallIntegerField(IntegerField):
- 小整數 -32768 ~ 32767
PositiveSmallIntegerField(PositiveIntegerRelDbTypeMixin, IntegerField)
- 正小整數 0 ~ 32767
IntegerField(Field)
- 整數列(有符号的) -2147483648 ~ 2147483647
PositiveIntegerField(PositiveIntegerRelDbTypeMixin, IntegerField)
- 正整數 0 ~ 2147483647
BigIntegerField(IntegerField):
- 長整型(有符号的) -9223372036854775808 ~ 9223372036854775807
自定義無符号整數字段
class UnsignedIntegerField(models.IntegerField):
def db_type(self, connection):
return 'integer UNSIGNED'
PS: 傳回值為字段在資料庫中的屬性,Django字段預設的值為:
'AutoField': 'integer AUTO_INCREMENT',
'BigAutoField': 'bigint AUTO_INCREMENT',
'BinaryField': 'longblob',
'BooleanField': 'bool',
'CharField': 'varchar(%(max_length)s)',
'CommaSeparatedIntegerField': 'varchar(%(max_length)s)',
'DateField': 'date',
'DateTimeField': 'datetime',
'DecimalField': 'numeric(%(max_digits)s, %(decimal_places)s)',
'DurationField': 'bigint',
'FileField': 'varchar(%(max_length)s)',
'FilePathField': 'varchar(%(max_length)s)',
'FloatField': 'double precision',
'IntegerField': 'integer',
'BigIntegerField': 'bigint',
'IPAddressField': 'char(15)',
'GenericIPAddressField': 'char(39)',
'NullBooleanField': 'bool',
'OneToOneField': 'integer',
'PositiveIntegerField': 'integer UNSIGNED',
'PositiveSmallIntegerField': 'smallint UNSIGNED',
'SlugField': 'varchar(%(max_length)s)',
'SmallIntegerField': 'smallint',
'TextField': 'longtext',
'TimeField': 'time',
'UUIDField': 'char(32)',
BooleanField(Field)
- 布爾值類型
NullBooleanField(Field):
- 可以為空的布爾值
CharField(Field)
- 字元類型
- 必須提供max_length參數, max_length表示字元長度
TextField(Field)
- 文本類型
EmailField(CharField):
- 字元串類型,Django Admin以及ModelForm中提供驗證機制
IPAddressField(Field)
- 字元串類型,Django Admin以及ModelForm中提供驗證 IPV4 機制
GenericIPAddressField(Field)
- 字元串類型,Django Admin以及ModelForm中提供驗證 Ipv4和Ipv6
- 參數:
protocol,用于指定Ipv4或Ipv6, 'both',"ipv4","ipv6"
unpack_ipv4, 如果指定為True,則輸入::ffff:192.0.2.1時候,可解析為192.0.2.1,開啟刺功能,需要protocol="both"
URLField(CharField)
- 字元串類型,Django Admin以及ModelForm中提供驗證 URL
SlugField(CharField)
- 字元串類型,Django Admin以及ModelForm中提供驗證支援 字母、數字、下劃線、連接配接符(減号)
CommaSeparatedIntegerField(CharField)
- 字元串類型,格式必須為逗号分割的數字
UUIDField(Field)
- 字元串類型,Django Admin以及ModelForm中提供對UUID格式的驗證
FilePathField(Field)
- 字元串,Django Admin以及ModelForm中提供讀取檔案夾下檔案的功能
- 參數:
path, 檔案夾路徑
match=None, 正則比對
recursive=False, 遞歸下面的檔案夾
allow_files=True, 允許檔案
allow_folders=False, 允許檔案夾
FileField(Field)
- 字元串,路徑儲存在資料庫,檔案上傳到指定目錄
- 參數:
upload_to = "" 上傳檔案的儲存路徑
storage = None 存儲元件,預設django.core.files.storage.FileSystemStorage
ImageField(FileField)
- 字元串,路徑儲存在資料庫,檔案上傳到指定目錄
- 參數:
upload_to = "" 上傳檔案的儲存路徑
storage = None 存儲元件,預設django.core.files.storage.FileSystemStorage
width_field=None, 上傳圖檔的高度儲存的資料庫字段名(字元串)
height_field=None 上傳圖檔的寬度儲存的資料庫字段名(字元串)
DateTimeField(DateField)
- 日期+時間格式 YYYY-MM-DD HH:MM[:ss[.uuuuuu]][TZ]
DateField(DateTimeCheckMixin, Field)
- 日期格式 YYYY-MM-DD
TimeField(DateTimeCheckMixin, Field)
- 時間格式 HH:MM[:ss[.uuuuuu]]
DurationField(Field)
- 長整數,時間間隔,資料庫中按照bigint存儲,ORM中擷取的值為datetime.timedelta類型
FloatField(Field)
- 浮點型
DecimalField(Field)
- 10進制小數
- 參數:
max_digits,小數總長度
decimal_places,小數位長度
BinaryField(Field)
- 二進制類型
字段
null 資料庫中字段是否可以為空
db_column 資料庫中字段的列名
db_tablespace
default 資料庫中字段的預設值
primary_key 資料庫中字段是否為主鍵
db_index 資料庫中字段是否可以建立索引
unique 資料庫中字段是否可以建立唯一索引
unique_for_date 資料庫中字段【日期】部分是否可以建立唯一索引
unique_for_month 資料庫中字段【月】部分是否可以建立唯一索引
unique_for_year 資料庫中字段【年】部分是否可以建立唯一索引
verbose_name Admin中顯示的字段名稱
blank Admin中是否允許使用者輸入為空
editable Admin中是否可以編輯
help_text Admin中該字段的提示資訊
choices Admin中顯示選擇框的内容,用不變動的資料放在記憶體中進而避免跨表操作
如:gf = models.IntegerField(choices=[(0, '何穗'),(1, '大表姐'),],default=1)
error_messages 自定義錯誤資訊(字典類型),進而定制想要顯示的錯誤資訊;
字典健:null, blank, invalid, invalid_choice, unique, and unique_for_date
如:{'null': "不能為空.", 'invalid': '格式錯誤'}
validators 自定義錯誤驗證(清單類型),進而定制想要的驗證規則
from django.core.validators import RegexValidator
from django.core.validators import EmailValidator,URLValidator,DecimalValidator,\
MaxLengthValidator,MinLengthValidator,MaxValueValidator,MinValueValidator
如:
test = models.CharField(
max_length=32,
error_messages={
'c1': '優先錯資訊1',
'c2': '優先錯資訊2',
'c3': '優先錯資訊3',
},
validators=[
RegexValidator(regex='root_\d+', message='錯誤了', code='c1'),
RegexValidator(regex='root_112233\d+', message='又錯誤了', code='c2'),
EmailValidator(message='又錯誤了', code='c3'), ]
)
參數
class UserInfo(models.Model):
nid = models.AutoField(primary_key=True)
username = models.CharField(max_length=32)
class Meta:
# 資料庫中生成的表名稱 預設 app名稱 + 下劃線 + 類名
db_table = "table_name"
# 聯合索引
index_together = [
("pub_date", "deadline"),
]
# 聯合唯一索引
unique_together = (("driver", "restaurant"),)
# admin中顯示的表名稱
verbose_name
# verbose_name加s
verbose_name_plural
更多:https://docs.djangoproject.com/en/1.10/ref/models/options/
元資訊
1.觸發Model中的驗證和錯誤提示有兩種方式:
a. Django Admin中的錯誤資訊會優先根據Admiin内部的ModelForm錯誤資訊提示,如果都成功,才來檢查Model的字段并顯示指定錯誤資訊
b. 調用Model對象的 clean_fields 方法,如:
# models.py
class UserInfo(models.Model):
nid = models.AutoField(primary_key=True)
username = models.CharField(max_length=32)
email = models.EmailField(error_messages={'invalid': '格式錯了.'})
# views.py
def index(request):
obj = models.UserInfo(username='11234', email='uu')
try:
print(obj.clean_fields())
except Exception as e:
print(e)
return HttpResponse('ok')
# Model的clean方法是一個鈎子,可用于定制操作,如:上述的異常處理。
2.Admin中修改錯誤提示
# admin.py
from django.contrib import admin
from model_club import models
from django import forms
class UserInfoForm(forms.ModelForm):
username = forms.CharField(error_messages={'required': '使用者名不能為空.'})
email = forms.EmailField(error_messages={'invalid': '郵箱格式錯誤.'})
age = forms.IntegerField(initial=1, error_messages={'required': '請輸入數值.', 'invalid': '年齡必須為數值.'})
class Meta:
model = models.UserInfo
# fields = ('username',)
fields = "__all__"
class UserInfoAdmin(admin.ModelAdmin):
form = UserInfoForm
admin.site.register(models.UserInfo, UserInfoAdmin)
拓展知識
2、連表結構
- 一對多:models.ForeignKey(其他表)
- 多對多:models.ManyToManyField(其他表)
- 一對一:models.OneToOneField(其他表)
應用場景:
一對多:當一張表中建立一行資料時,有一個單選的下拉框(可以被重複選擇)
例如:建立使用者資訊時候,需要選擇一個使用者類型【普通使用者】【金牌使用者】【鉑金使用者】等。
多對多:在某表中建立一行資料是,有一個可以多選的下拉框
例如:建立使用者資訊,需要為使用者指定多個愛好
一對一:在某表中建立一行資料時,有一個單選的下拉框(下拉框中的内容被用過一次就消失了
例如:原有含10列資料的一張表儲存相關資訊,經過一段時間之後,10列無法滿足需求,需要為原來的表再添加5列資料
ForeignKey(ForeignObject) # ForeignObject(RelatedField)
to, # 要進行關聯的表名
to_field=None, # 要關聯的表中的字段名稱
on_delete=None, # 當删除關聯表中的資料時,目前表與其關聯的行的行為
- models.CASCADE,删除關聯資料,與之關聯也删除
- models.DO_NOTHING,删除關聯資料,引發錯誤IntegrityError
- models.PROTECT,删除關聯資料,引發錯誤ProtectedError
- models.SET_NULL,删除關聯資料,與之關聯的值設定為null(前提FK字段需要設定為可空)
- models.SET_DEFAULT,删除關聯資料,與之關聯的值設定為預設值(前提FK字段需要設定預設值)
- models.SET,删除關聯資料,
a. 與之關聯的值設定為指定值,設定:models.SET(值)
b. 與之關聯的值設定為可執行對象的傳回值,設定:models.SET(可執行對象)
def func():
return 10
class MyModel(models.Model):
user = models.ForeignKey(
to="User",
to_field="id"
on_delete=models.SET(func),)
related_name=None, # 反向操作時,使用的字段名,用于代替 【表名_set】 如: obj.表名_set.all()
related_query_name=None, # 反向操作時,使用的連接配接字首,用于替換【表名】 如: models.UserGroup.objects.filter(表名__字段名=1).values('表名__字段名')
limit_choices_to=None, # 在Admin或ModelForm中顯示關聯資料時,提供的條件:
# 如:
- limit_choices_to={'nid__gt': 5}
- limit_choices_to=lambda : {'nid__gt': 5}
from django.db.models import Q
- limit_choices_to=Q(nid__gt=10)
- limit_choices_to=Q(nid=8) | Q(nid__gt=10)
- limit_choices_to=lambda : Q(Q(nid=8) | Q(nid__gt=10)) & Q(caption='root')
db_constraint=True # 是否在資料庫中建立外鍵限制
parent_link=False # 在Admin中是否顯示關聯資料
OneToOneField(ForeignKey)
to, # 要進行關聯的表名
to_field=None # 要關聯的表中的字段名稱
on_delete=None, # 當删除關聯表中的資料時,目前表與其關聯的行的行為
###### 對于一對一 ######
# 1. 一對一其實就是 一對多 + 唯一索引
# 2.當兩個類之間有繼承關系時,預設會建立一個一對一字段
# 如下會在A表中額外增加一個c_ptr_id列且唯一:
class C(models.Model):
nid = models.AutoField(primary_key=True)
part = models.CharField(max_length=12)
class A(C):
id = models.AutoField(primary_key=True)
code = models.CharField(max_length=1)
ManyToManyField(RelatedField)
to, # 要進行關聯的表名
related_name=None, # 反向操作時,使用的字段名,用于代替 【表名_set】 如: obj.表名_set.all()
related_query_name=None, # 反向操作時,使用的連接配接字首,用于替換【表名】 如: models.UserGroup.objects.filter(表名__字段名=1).values('表名__字段名')
limit_choices_to=None, # 在Admin或ModelForm中顯示關聯資料時,提供的條件:
# 如:
- limit_choices_to={'nid__gt': 5}
- limit_choices_to=lambda : {'nid__gt': 5}
from django.db.models import Q
- limit_choices_to=Q(nid__gt=10)
- limit_choices_to=Q(nid=8) | Q(nid__gt=10)
- limit_choices_to=lambda : Q(Q(nid=8) | Q(nid__gt=10)) & Q(caption='root')
symmetrical=None, # 僅用于多對多自關聯時,symmetrical用于指定内部是否建立反向操作的字段
# 做如下操作時,不同的symmetrical會有不同的可選字段
models.BB.objects.filter(...)
# 可選字段有:code, id, m1
class BB(models.Model):
code = models.CharField(max_length=12)
m1 = models.ManyToManyField('self',symmetrical=True)
# 可選字段有: bb, code, id, m1
class BB(models.Model):
code = models.CharField(max_length=12)
m1 = models.ManyToManyField('self',symmetrical=False)
through=None, # 自定義第三張表時,使用字段用于指定關系表
through_fields=None, # 自定義第三張表時,使用字段用于指定關系表中那些字段做多對多關系表
from django.db import models
class Person(models.Model):
name = models.CharField(max_length=50)
class Group(models.Model):
name = models.CharField(max_length=128)
members = models.ManyToManyField(
Person,
through='Membership',
through_fields=('group', 'person'),
)
class Membership(models.Model):
group = models.ForeignKey(Group, on_delete=models.CASCADE)
person = models.ForeignKey(Person, on_delete=models.CASCADE)
inviter = models.ForeignKey(
Person,
on_delete=models.CASCADE,
related_name="membership_invites",
)
invite_reason = models.CharField(max_length=64)
db_constraint=True, # 是否在資料庫中建立外鍵限制
db_table=None, # 預設建立第三張表時,資料庫中表的名稱
字段以及參數
二、操作表
1、基本操作
# 增
#
# models.Tb1.objects.create(c1='xx', c2='oo') 增加一條資料,可以接受字典類型資料 **kwargs
# obj = models.Tb1(c1='xx', c2='oo')
# obj.save()
# 查
#
# models.Tb1.objects.get(id=123) # 擷取單條資料,不存在則報錯(不建議)
# models.Tb1.objects.all() # 擷取全部
# models.Tb1.objects.filter(name='seven') # 擷取指定條件的資料
# 删
#
# models.Tb1.objects.filter(name='seven').delete() # 删除指定條件的資料
# 改
# models.Tb1.objects.filter(name='seven').update(gender='0') # 将指定條件的資料更新,均支援 **kwargs
# obj = models.Tb1.objects.get(id=1)
# obj.c1 = '111'
# obj.save() # 修改單條資料
基本操作
2、進階操作(了不起的雙下劃線)
利用雙下劃線将字段和對應的操作連接配接起來
# 擷取個數
#
# models.Tb1.objects.filter(name='seven').count()
# 大于,小于
#
# models.Tb1.objects.filter(id__gt=1) # 擷取id大于1的值
# models.Tb1.objects.filter(id__gte=1) # 擷取id大于等于1的值
# models.Tb1.objects.filter(id__lt=10) # 擷取id小于10的值
# models.Tb1.objects.filter(id__lte=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的資料
# models.Tb1.objects.exclude(id__in=[11, 22, 33]) # not in
# isnull
# Entry.objects.filter(pub_date__isnull=True)
# contains
#
# models.Tb1.objects.filter(name__contains="ven")
# models.Tb1.objects.filter(name__icontains="ven") # icontains大小寫不敏感
# models.Tb1.objects.exclude(name__icontains="ven")
# range
#
# models.Tb1.objects.filter(id__range=[1, 2]) # 範圍bettwen and
# 其他類似
#
# 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
# 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"
# limit 、offset
#
# models.Tb1.objects.all()[10:20]
# regex正則比對,iregex 不區分大小寫
#
# Entry.objects.get(title__regex=r'^(An?|The) +')
# Entry.objects.get(title__iregex=r'^(an?|the) +')
# date
#
# Entry.objects.filter(pub_date__date=datetime.date(2005, 1, 1))
# Entry.objects.filter(pub_date__date__gt=datetime.date(2005, 1, 1))
# year
#
# Entry.objects.filter(pub_date__year=2005)
# Entry.objects.filter(pub_date__year__gte=2005)
# month
#
# Entry.objects.filter(pub_date__month=12)
# Entry.objects.filter(pub_date__month__gte=6)
# day
#
# Entry.objects.filter(pub_date__day=3)
# Entry.objects.filter(pub_date__day__gte=3)
# week_day
#
# Entry.objects.filter(pub_date__week_day=2)
# Entry.objects.filter(pub_date__week_day__gte=2)
# hour
#
# Event.objects.filter(timestamp__hour=23)
# Event.objects.filter(time__hour=5)
# Event.objects.filter(timestamp__hour__gte=12)
# minute
#
# Event.objects.filter(timestamp__minute=29)
# Event.objects.filter(time__minute=46)
# Event.objects.filter(timestamp__minute__gte=29)
# second
#
# Event.objects.filter(timestamp__second=31)
# Event.objects.filter(time__second=2)
# Event.objects.filter(timestamp__second__gte=31)
進階操作
3、其他操作
# extra
#
# extra(self, select=None, where=None, params=None, tables=None, order_by=None, select_params=None)
# Entry.objects.extra(select={'new_id': "select col from sometable where othercol > %s"}, select_params=(1,))
# Entry.objects.extra(where=['headline=%s'], params=['Lennon'])
# Entry.objects.extra(where=["foo='a' OR bar = 'a'", "baz = 'a'"])
# Entry.objects.extra(select={'new_id': "select id from tb where id > %s"}, select_params=(1,), order_by=['-nid'])
# F
#
# from django.db.models import F
# models.Tb1.objects.update(num=F('num')+1)
# Q
#
# 方式一:
# Q(nid__gt=10)
# Q(nid=8) | Q(nid__gt=10)
# Q(Q(nid=8) | Q(nid__gt=10)) & Q(caption='root')
# 方式二:
# con = Q()
# q1 = Q()
# q1.connector = 'OR'
# q1.children.append(('id', 1))
# q1.children.append(('id', 10))
# q1.children.append(('id', 9))
# q2 = Q()
# q2.connector = 'OR'
# q2.children.append(('c1', 1))
# q2.children.append(('c1', 10))
# q2.children.append(('c1', 9))
# con.add(q1, 'AND')
# con.add(q2, 'AND')
#
# models.Tb1.objects.filter(con)
# 執行原生SQL
#
# from django.db import connection, connections
# cursor = connection.cursor() # cursor = connections['default'].cursor()
# cursor.execute("""SELECT * from auth_user where id = %s""", [1])
# row = cursor.fetchone()
其他操作
4、連表操作(了不起的雙下劃線)
利用雙下劃線和 _set 将表之間的操作連接配接起來
class UserProfile(models.Model):
user_info = models.OneToOneField('UserInfo')
username = models.CharField(max_length=64)
password = models.CharField(max_length=64)
def __unicode__(self):
return self.username
class UserInfo(models.Model):
user_type_choice = (
(0, u'普通使用者'),
(1, u'進階使用者'),
)
user_type = models.IntegerField(choices=user_type_choice)
name = models.CharField(max_length=32)
email = models.CharField(max_length=32)
address = models.CharField(max_length=128)
def __unicode__(self):
return self.name
class UserGroup(models.Model):
caption = models.CharField(max_length=64)
user_info = models.ManyToManyField('UserInfo')
def __unicode__(self):
return self.caption
class Host(models.Model):
hostname = models.CharField(max_length=64)
ip = models.GenericIPAddressField()
user_group = models.ForeignKey('UserGroup')
def __unicode__(self):
return self.hostname
表結構執行個體
user_info_obj = models.UserInfo.objects.filter(id=1).first()
print user_info_obj.user_type
print user_info_obj.get_user_type_display()
print user_info_obj.userprofile.password
user_info_obj = models.UserInfo.objects.filter(id=1).values('email', 'userprofile__username').first()
print user_info_obj.keys()
print user_info_obj.values()
一對一操作
類似一對一
1、搜尋條件使用 __ 連接配接
2、擷取值時使用 . 連接配接
一對多
user_info_obj = models.UserInfo.objects.get(name=u'武沛齊')
user_info_objs = models.UserInfo.objects.all()
group_obj = models.UserGroup.objects.get(caption='CEO')
group_objs = models.UserGroup.objects.all()
# 添加資料
#group_obj.user_info.add(user_info_obj)
#group_obj.user_info.add(*user_info_objs)
# 删除資料
#group_obj.user_info.remove(user_info_obj)
#group_obj.user_info.remove(*user_info_objs)
# 添加資料
#user_info_obj.usergroup_set.add(group_obj)
#user_info_obj.usergroup_set.add(*group_objs)
# 删除資料
#user_info_obj.usergroup_set.remove(group_obj)
#user_info_obj.usergroup_set.remove(*group_objs)
# 擷取資料
#print group_obj.user_info.all()
#print group_obj.user_info.all().filter(id=1)
# 擷取資料
#print user_info_obj.usergroup_set.all()
#print user_info_obj.usergroup_set.all().filter(caption='CEO')
#print user_info_obj.usergroup_set.all().filter(caption='DBA')
多對多操作
擴充:
a、自定義上傳
def upload_file(request):
if request.method == "POST":
obj = request.FILES.get('fafafa')
f = open(obj.name, 'wb')
for chunk in obj.chunks():
f.write(chunk)
f.close()
return render(request, 'file.html')
View Code
b、Form上傳檔案執行個體
class FileForm(forms.Form):
ExcelFile = forms.FileField()
Form
from django.db import models
class UploadFile(models.Model):
userid = models.CharField(max_length = 30)
file = models.FileField(upload_to = './upload/')
date = models.DateTimeField(auto_now_add=True)
Model
def UploadFile(request):
uf = AssetForm.FileForm(request.POST,request.FILES)
if uf.is_valid():
upload = models.UploadFile()
upload.userid = 1
upload.file = uf.cleaned_data['ExcelFile']
upload.save()
print upload.file
View
Form
django中的Form一般有兩種功能:
- 輸入html
- 驗證使用者輸入
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import re
from django import forms
from django.core.exceptions import ValidationError
def mobile_validate(value):
mobile_re = re.compile(r'^(13[0-9]|15[012356789]|17[678]|18[0-9]|14[57])[0-9]{8}$')
if not mobile_re.match(value):
raise ValidationError('手機号碼格式錯誤')
class PublishForm(forms.Form):
user_type_choice = (
(0, u'普通使用者'),
(1, u'進階使用者'),
)
user_type = forms.IntegerField(widget=forms.widgets.Select(choices=user_type_choice,
attrs={'class': "form-control"}))
title = forms.CharField(max_length=20,
min_length=5,
error_messages={'required': u'标題不能為空',
'min_length': u'标題最少為5個字元',
'max_length': u'标題最多為20個字元'},
widget=forms.TextInput(attrs={'class': "form-control",
'placeholder': u'标題5-20個字元'}))
memo = forms.CharField(required=False,
max_length=256,
widget=forms.widgets.Textarea(attrs={'class': "form-control no-radius", 'placeholder': u'較長的描述', 'rows': 3}))
phone = forms.CharField(validators=[mobile_validate, ],
error_messages={'required': u'手機不能為空'},
widget=forms.TextInput(attrs={'class': "form-control",
'placeholder': u'手機号碼'}))
email = forms.EmailField(required=False,
error_messages={'required': u'郵箱不能為空','invalid': u'郵箱格式錯誤'},
widget=forms.TextInput(attrs={'class': "form-control", 'placeholder': u'郵箱'}))
Form
def publish(request):
ret = {'status': False, 'data': '', 'error': '', 'summary': ''}
if request.method == 'POST':
request_form = PublishForm(request.POST)
if request_form.is_valid():
request_dict = request_form.clean()
print request_dict
ret['status'] = True
else:
error_msg = request_form.errors.as_json()
ret['error'] = json.loads(error_msg)
return HttpResponse(json.dumps(ret))
View
擴充:ModelForm
在使用Model和Form時,都需要對字段進行定義并指定類型,通過ModelForm則可以省去From中字段的定義
class AdminModelForm(forms.ModelForm):
class Meta:
model = models.Admin
#fields = '__all__'
fields = ('username', 'email')
widgets = {
'email' : forms.PasswordInput(attrs={'class':"alex"}),
}
View Code
跨站請求僞造
一、簡介
django為使用者實作防止跨站請求僞造的功能,通過中間件 django.middleware.csrf.CsrfViewMiddleware 來完成。而對于django中設定防跨站請求僞造功能有分為全局和局部。
全局:
中間件 django.middleware.csrf.CsrfViewMiddleware
局部:
- @csrf_protect,為目前函數強制設定防跨站請求僞造功能,即便settings中沒有設定全局中間件。
- @csrf_exempt,取消目前函數防跨站請求僞造功能,即便settings中設定了全局中間件。
注:from django.views.decorators.csrf import csrf_exempt,csrf_protect
二、應用
1、普通表單
+ View Code ?
1 2 3 4 5 6 7 | |
2、Ajax
對于傳統的form,可以通過表單的方式将token再次發送到服務端,而對于ajax的話,使用如下方式。
view.py
+ View Code ?
1 2 3 4 5 6 7 8 9 10 | |
text.html
+ View Code ?
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 | |
更多:https://docs.djangoproject.com/en/dev/ref/csrf/#ajax
Cookie
1、擷取Cookie:
?
1 2 3 4 5 6 | |
2、設定Cookie:
?
1 2 3 4 5 6 7 8 9 10 11 12 13 | |
由于cookie儲存在用戶端的電腦上,是以,JavaScript和jquery也可以操作cookie。
?
1 2 | |
Session
Django中預設支援Session,其内部提供了5種類型的Session供開發者使用:
- 資料庫(預設)
- 緩存
- 檔案
- 緩存+資料庫
- 加密cookie
1、資料庫Session
+ View Code ?
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 | |
2、緩存Session
+ View Code ?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | |
3、檔案Session
+ View Code ?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | |
4、緩存+資料庫Session
+ View Code ?
1 2 3 4 5 6 7 8 9 | |
5、加密cookie Session
+ View Code ?
1 2 3 4 5 6 7 | |
更多參考:猛擊這裡 和 猛擊這裡
擴充:Session使用者驗證
?
1 2 3 4 5 6 7 | |
分頁
一、Django内置分頁
from django.shortcuts import render
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
L = []
for i in range(999):
L.append(i)
def index(request):
current_page = request.GET.get('p')
paginator = Paginator(L, 10)
# per_page: 每頁顯示條目數量
# count: 資料總個數
# num_pages:總頁數
# page_range:總頁數的索引範圍,如: (1,10),(1,200)
# page: page對象
try:
posts = paginator.page(current_page)
# has_next 是否有下一頁
# next_page_number 下一頁頁碼
# has_previous 是否有上一頁
# previous_page_number 上一頁頁碼
# object_list 分頁之後的資料清單
# number 目前頁
# paginator paginator對象
except PageNotAnInteger:
posts = paginator.page(1)
except EmptyPage:
posts = paginator.page(paginator.num_pages)
return render(request, 'index.html', {'posts': posts})
views.py
<!DOCTYPE html>
<html>
<head en">
<meta charset="UTF-8">
<title></title>
</head>
<body>
<ul>
{% for item in posts %}
<li>{{ item }}</li>
{% endfor %}
</ul>
<div class="pagination">
<span class="step-links">
{% if posts.has_previous %}
<a href="?p={{ posts.previous_page_number }}">Previous</a>
{% endif %}
<span class="current">
Page {{ posts.number }} of {{ posts.paginator.num_pages }}.
</span>
{% if posts.has_next %}
<a href="?p={{ posts.next_page_number }}">Next</a>
{% endif %}
</span>
</div>
</body>
</html>
Html
from django.shortcuts import render
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
class CustomPaginator(Paginator):
def __init__(self, current_page, max_pager_num, *args, **kwargs):
"""
:param current_page: 目前頁
:param max_pager_num:最多顯示的頁碼個數
:param args:
:param kwargs:
:return:
"""
self.current_page = int(current_page)
self.max_pager_num = max_pager_num
super(CustomPaginator, self).__init__(*args, **kwargs)
def page_num_range(self):
# 目前頁面
# self.current_page
# 總頁數
# self.num_pages
# 最多顯示的頁碼個數
# self.max_pager_num
print(1)
if self.num_pages < self.max_pager_num:
return range(1, self.num_pages + 1)
print(2)
part = int(self.max_pager_num / 2)
if self.current_page - part < 1:
return range(1, self.max_pager_num + 1)
print(3)
if self.current_page + part > self.num_pages:
return range(self.num_pages + 1 - self.max_pager_num, self.num_pages + 1)
print(4)
return range(self.current_page - part, self.current_page + part + 1)
L = []
for i in range(999):
L.append(i)
def index(request):
current_page = request.GET.get('p')
paginator = CustomPaginator(current_page, 11, L, 10)
# per_page: 每頁顯示條目數量
# count: 資料總個數
# num_pages:總頁數
# page_range:總頁數的索引範圍,如: (1,10),(1,200)
# page: page對象
try:
posts = paginator.page(current_page)
# has_next 是否有下一頁
# next_page_number 下一頁頁碼
# has_previous 是否有上一頁
# previous_page_number 上一頁頁碼
# object_list 分頁之後的資料清單
# number 目前頁
# paginator paginator對象
except PageNotAnInteger:
posts = paginator.page(1)
except EmptyPage:
posts = paginator.page(paginator.num_pages)
return render(request, 'index.html', {'posts': posts})
擴充内置分頁:views.py
<!DOCTYPE html>
<html>
<head en">
<meta charset="UTF-8">
<title></title>
</head>
<body>
<ul>
{% for item in posts %}
<li>{{ item }}</li>
{% endfor %}
</ul>
<div class="pagination">
<span class="step-links">
{% if posts.has_previous %}
<a href="?p={{ posts.previous_page_number }}">Previous</a>
{% endif %}
{% for i in posts.paginator.page_num_range %}
<a href="?p={{ i }}">{{ i }}</a>
{% endfor %}
{% if posts.has_next %}
<a href="?p={{ posts.next_page_number }}">Next</a>
{% endif %}
</span>
<span class="current">
Page {{ posts.number }} of {{ posts.paginator.num_pages }}.
</span>
</div>
</body>
</html>
擴充内置分頁:Html
二、自定義分頁
分頁功能在每個網站都是必要的,對于分頁來說,其實就是根據使用者的輸入計算出應該在資料庫表中的起始位置。
1、設定每頁顯示資料條數
2、使用者輸入頁碼(第一頁、第二頁...)
3、根據設定的每頁顯示條數和目前頁碼,計算出需要取資料表的起始位置
4、在資料表中根據起始位置取值,頁面上輸出資料
需求又來了,需要在頁面上顯示分頁的頁面。如:[上一頁][1][2][3][4][5][下一頁]
1、設定每頁顯示資料條數
2、使用者輸入頁碼(第一頁、第二頁...)
3、設定顯示多少頁号
4、擷取目前資料總條數
5、根據設定顯示多少頁号和資料總條數計算出,總頁數
6、根據設定的每頁顯示條數和目前頁碼,計算出需要取資料表的起始位置
7、在資料表中根據起始位置取值,頁面上輸出資料
8、輸出分頁html,如:[上一頁][1][2][3][4][5][下一頁]
#!/usr/bin/env python
# _*_coding:utf-8_*_
from django.utils.safestring import mark_safe
class PageInfo(object):
def __init__(self,current,totalItem,peritems=5):
self.__current=current
self.__peritems=peritems
self.__totalItem=totalItem
def From(self):
return (self.__current-1)*self.__peritems
def To(self):
return self.__current*self.__peritems
def TotalPage(self): #總頁數
result=divmod(self.__totalItem,self.__peritems)
if result[1]==0:
return result[0]
else:
return result[0]+1
def Custompager(baseurl,currentPage,totalpage): #基礎頁,目前頁,總頁數
perPager=11
#總頁數<11
#0 -- totalpage
#總頁數>11
#目前頁大于5 currentPage-5 -- currentPage+5
#currentPage+5是否超過總頁數,超過總頁數,end就是總頁數
#目前頁小于5 0 -- 11
begin=0
end=0
if totalpage <= 11:
begin=0
end=totalpage
else:
if currentPage>5:
begin=currentPage-5
end=currentPage+5
if end > totalpage:
end=totalpage
else:
begin=0
end=11
pager_list=[]
if currentPage<=1:
first="<a href=''>首頁</a>"
else:
first="<a href='%s%d'>首頁</a>" % (baseurl,1)
pager_list.append(first)
if currentPage<=1:
prev="<a href=''>上一頁</a>"
else:
prev="<a href='%s%d'>上一頁</a>" % (baseurl,currentPage-1)
pager_list.append(prev)
for i in range(begin+1,end+1):
if i == currentPage:
temp="<a href='%s%d' class='selected'>%d</a>" % (baseurl,i,i)
else:
temp="<a href='%s%d'>%d</a>" % (baseurl,i,i)
pager_list.append(temp)
if currentPage>=totalpage:
next="<a href='#'>下一頁</a>"
else:
next="<a href='%s%d'>下一頁</a>" % (baseurl,currentPage+1)
pager_list.append(next)
if currentPage>=totalpage:
last="<a href=''>末頁</a>"
else:
last="<a href='%s%d'>末頁</a>" % (baseurl,totalpage)
pager_list.append(last)
result=''.join(pager_list)
return mark_safe(result) #把字元串轉成html語言
分頁執行個體
總結,分頁時需要做三件事:
- 建立處理分頁資料的類
- 根據分頁資料擷取資料
- 輸出分頁HTML,即:[上一頁][1][2][3][4][5][下一頁]
緩存
由于Django是動态網站,所有每次請求均會去資料進行相應的操作,當程式通路量大時,耗時必然會更加明顯,最簡單解決方式是使用:緩存,緩存将一個某個views的傳回值儲存至記憶體或者memcache中,5分鐘内再有人來通路時,則不再去執行view中的操作,而是直接從記憶體或者Redis中之前緩存的内容拿到,并傳回。
Django中提供了6種緩存方式:
- 開發調試
- 記憶體
- 檔案
- 資料庫
- Memcache緩存(python-memcached子產品)
- Memcache緩存(pylibmc子產品)
1、配置
a、開發調試
# 此為開始調試用,實際内部不做任何操作
# 配置:
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.dummy.DummyCache', # 引擎
'TIMEOUT': 300, # 緩存逾時時間(預設300,None表示永不過期,0表示立即過期)
'OPTIONS':{
'MAX_ENTRIES': 300, # 最大緩存個數(預設300)
'CULL_FREQUENCY': 3, # 緩存到達最大個數之後,剔除緩存個數的比例,即:1/CULL_FREQUENCY(預設3)
},
'KEY_PREFIX': '', # 緩存key的字首(預設空)
'VERSION': 1, # 緩存key的版本(預設1)
'KEY_FUNCTION' 函數名 # 生成key的函數(預設函數會生成為:【字首:版本:key】)
}
}
# 自定義key
def default_key_func(key, key_prefix, version):
"""
Default function to generate keys.
Constructs the key used by all other methods. By default it prepends
the `key_prefix'. KEY_FUNCTION can be used to specify an alternate
function with custom key making behavior.
"""
return '%s:%s:%s' % (key_prefix, version, key)
def get_key_func(key_func):
"""
Function to decide which key function to use.
Defaults to ``default_key_func``.
"""
if key_func is not None:
if callable(key_func):
return key_func
else:
return import_string(key_func)
return default_key_func
View Code
b、記憶體
# 此緩存将内容儲存至記憶體的變量中
# 配置:
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
'LOCATION': 'unique-snowflake',
}
}
# 注:其他配置同開發調試版本
View Code
c、檔案
# 此緩存将内容儲存至檔案
# 配置:
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache',
'LOCATION': '/var/tmp/django_cache',
}
}
# 注:其他配置同開發調試版本
View Code
d、資料庫
# 此緩存将内容儲存至資料庫
# 配置:
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.db.DatabaseCache',
'LOCATION': 'my_cache_table', # 資料庫表
}
}
# 注:執行建立表指令 python manage.py createcachetable
View Code
e、Memcache緩存(python-memcached子產品)
# 此緩存使用python-memcached子產品連接配接memcache
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache',
'LOCATION': '127.0.0.1:11211',
}
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache',
'LOCATION': 'unix:/tmp/memcached.sock',
}
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache',
'LOCATION': [
'172.19.26.240:11211',
'172.19.26.242:11211',
]
}
}
View Code
f、Memcache緩存(pylibmc子產品)
# 此緩存使用pylibmc子產品連接配接memcache
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache',
'LOCATION': '127.0.0.1:11211',
}
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache',
'LOCATION': '/tmp/memcached.sock',
}
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache',
'LOCATION': [
'172.19.26.240:11211',
'172.19.26.242:11211',
]
}
}
View Code
g. Redis緩存(依賴:pip3 install django-redis)
CACHES = {
"default": {
"BACKEND": "django_redis.cache.RedisCache",
"LOCATION": "redis://127.0.0.1:6379",
"OPTIONS": {
"CLIENT_CLASS": "django_redis.client.DefaultClient",
"CONNECTION_POOL_KWARGS": {"max_connections": 100}
# "PASSWORD": "密碼",
}
}
}
View Code
from django_redis import get_redis_connection
conn = get_redis_connection("default")
視圖中連結并操作
2、應用
a. 全站使用
使用中間件,經過一系列的認證等操作,如果内容在緩存中存在,則使用FetchFromCacheMiddleware擷取内容并傳回給使用者,當傳回給使用者之前,判斷緩存中是否已經存在,如果不存在則UpdateCacheMiddleware會将緩存儲存至緩存,進而實作全站緩存
MIDDLEWARE = [
'django.middleware.cache.UpdateCacheMiddleware',
# 其他中間件...
'django.middleware.cache.FetchFromCacheMiddleware',
]
CACHE_MIDDLEWARE_ALIAS = ""
CACHE_MIDDLEWARE_SECONDS = ""
CACHE_MIDDLEWARE_KEY_PREFIX = ""
View Code
b. 單獨視圖緩存
方式一:
from django.views.decorators.cache import cache_page
@cache_page(60 * 15)
def my_view(request):
...
方式二:
from django.views.decorators.cache import cache_page
urlpatterns = [
url(r'^foo/([0-9]{1,2})/$', cache_page(60 * 15)(my_view)),
]
View Code
c、局部視圖使用
a. 引入TemplateTag
{% load cache %}
b. 使用緩存
{% cache 5000 緩存key %}
緩存内容
{% endcache %}
View Code
更多:猛擊這裡
序列化
關于Django中的序列化主要應用在将資料庫中檢索的資料傳回給用戶端使用者,特别的Ajax請求一般傳回的為Json格式。
1、serializers
?
1 2 3 4 5 | |
2、json.dumps
?
1 2 3 4 5 6 7 8 | |
由于json.dumps時無法處理datetime日期,是以可以通過自定義處理器來做擴充,如:
+ View Code ?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | |
信号
Django中提供了“信号排程”,用于在架構執行操作時解耦。通俗來講,就是一些動作發生的時候,信号允許特定的發送者去提醒一些接受者。
1、Django内置信号
?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | |
對于Django内置的信号,僅需注冊指定信号,當程式執行相應操作時,自動觸發注冊函數:
from django.core.signals import request_finished
from django.core.signals import request_started
from django.core.signals import got_request_exception
from django.db.models.signals import class_prepared
from django.db.models.signals import pre_init, post_init
from django.db.models.signals import pre_save, post_save
from django.db.models.signals import pre_delete, post_delete
from django.db.models.signals import m2m_changed
from django.db.models.signals import pre_migrate, post_migrate
from django.test.signals import setting_changed
from django.test.signals import template_rendered
from django.db.backends.signals import connection_created
def callback(sender, **kwargs):
print("xxoo_callback")
print(sender,kwargs)
xxoo.connect(callback)
# xxoo指上述導入的内容
View Code
from django.core.signals import request_finished
from django.dispatch import receiver
@receiver(request_finished)
def my_callback(sender, **kwargs):
print("Request finished!")
View Code
2、自定義信号
a. 定義信号
?
1 2 | |
b. 注冊信号
?
1 2 3 4 5 | |
c. 觸發信号
?
1 2 3 | |
由于内置信号的觸發者已經內建到Django中,是以其會自動調用,而對于自定義信号則需要開發者在任意位置觸發。
更多:猛擊這裡
轉載于:https://www.cnblogs.com/AbnerLc/p/11604467.html