Django?ORM?事務(wù)和查詢優(yōu)化的操作方法
一、事務(wù)操作
模塊 from django.db import transaction
1 開啟事務(wù):with transaction.atomic()
from django.db import transaction
class MyView(View):
def post(self, request):
# 在with代碼塊中開啟事務(wù),出了with自動(dòng)結(jié)束事務(wù)
with transaction.atomic()
pass開啟事務(wù):@transaction.atomic
class MyView(View):
@transaction.atomic
def post(self, request):
# 整個(gè)post方法都在事務(wù)中
pass2 設(shè)置事務(wù)回滾點(diǎn)
sid = transaction.savepoint() # 可以設(shè)置多個(gè)
事務(wù)回滾
# 回滾整個(gè)事務(wù) 但是不能在自動(dòng)提交的事務(wù)中使用 transaction.rollback() # 回滾到sid transaction.savepoint_rollback(sid)
3 事務(wù)提交
# 提交整個(gè)事務(wù) 但是不能在自動(dòng)提交的事務(wù)中使用 transaction.commit() # 釋放保存點(diǎn) sid ,自保存點(diǎn)被創(chuàng)建依賴執(zhí)行的更改成為事務(wù)的一部分 transaction.savepoint_commit(sid) # atomic的事務(wù),只有在事務(wù)結(jié)束后,才會自動(dòng)提交到數(shù)據(jù)庫,進(jìn)行數(shù)據(jù)的持久化
4 設(shè)置事務(wù)隔離
配置文件
import psycopg2.extensions
# 根據(jù)需要設(shè)置隔離級別,如下配置是READ_COMMITTED
# 但是配置PostgreSQL,不能配置mysql,否則會報(bào)錯(cuò)
DATABASES = {
# ...
'OPTIONS': {
'isolation_level': psycopg2.extensions.ISOLATION_LEVEL_READ_COMMITTED,
},
}二、 ORM 惰性查詢
即盡量減少對數(shù)據(jù)庫的操作,減輕數(shù)據(jù)庫的壓力
1 ORM惰性查詢
ORM 內(nèi)所有的 SQL 查詢語句,都是惰性操作,即當(dāng)你需要使用查詢結(jié)果時(shí),才會真正去執(zhí)行 SQL 語句訪問數(shù)據(jù)庫,否則是不會執(zhí)行查詢的
可使用代碼驗(yàn)證:
2 配置settings文件,使得當(dāng)執(zhí)行 SQL 語句時(shí),在終端打印 SQL 命令
LOGGING = {
'version': 1,
'disable_existing_loggers': False,
'handlers': {
'console': {
'level': 'DEBUG',
'class': 'logging.StreamHandler',
},
},
'loggers': {
'django.db.backends': {
'handlers': ['console'],
'propagate': True,
'level': 'DEBUG',
},
}
}3 驗(yàn)證
# 只查詢,不使用查詢結(jié)果 models.User.objects.all() # 將查詢結(jié)果賦值給變量res,并使用res res = models.User.objects.all() print(res) # 分別執(zhí)行,查看終端的打印結(jié)果
三、only與defer
only 查詢
- 指定查詢某字段時(shí),將全部結(jié)果一次性獲取出來,以后再訪問該字段數(shù)據(jù)時(shí),直接從內(nèi)存中拿即可,不需要再操作數(shù)據(jù)庫,返回QuerySet[obj1, obj2]
defer 查詢
- 指的是一次性獲取指定字段之外的所有字段數(shù)據(jù),封裝到對象中,以后再訪問這些字段數(shù)據(jù)時(shí),直接從內(nèi)存中拿即可,不需要再操作數(shù)據(jù)庫
class UserTest(models.Model):
id = models.AutoField(primary_key = True)
name = models.CharField(max_length = 32)
age = models.IntegerField()
def __str__(self):
return 'name: %s, age: %s' %(self.name, self.age)
models.UserTest.objects.create(name = '小慧',age = '18')
models.UserTest.objects.create(name = '小嘉',age = '19')
models.UserTest.objects.create(name = '小甜',age = '17')
models.UserTest.objects.create(name = '小妞',age = '16')
# all 查詢
# 訪問一次數(shù)據(jù)庫,打印一條sql命令,獲取所有的 UserTest 表中的數(shù)據(jù)
res = models.UserTest.objects.all()
print(res)
# only 查詢
# 訪問一次數(shù)據(jù)庫,打印一條sql命令,僅僅獲取 name 的所有數(shù)據(jù)到內(nèi)存
res = models.UserTest.objects.only("name")
print(res)
for user_obj in res:
print(user_obj.name) # 直接從內(nèi)存中獲取,不訪問數(shù)據(jù)庫,不打印 SQl 命令
print(user_obj.age) # 每執(zhí)行一次,就訪問一次數(shù)據(jù)庫,打印一次 SQl 命令
# defer 查詢
# 訪問一次數(shù)據(jù)庫,打印一條 SQl 命令,獲取除了 name 之外的所有數(shù)據(jù)到內(nèi)存
res = models.User.objects.defer("name")
for user_obj in res:
print(user_obj.name) # 每執(zhí)行一次,訪問一次數(shù)據(jù)庫,打印一條 SQl 命令
print(user_obj.age) # 直接從內(nèi)存中獲取,不訪問數(shù)據(jù)庫,不打印 SQl 命令注意:分別執(zhí)行才能看到效果 注釋print
四、select_related與prefetch_related
都支持正反向跨表查詢,規(guī)則等同于基于對象的正反向查詢,正向:'外鍵',反向:'表名小寫_set'
select_related:指定外鍵,連表查詢(inner join),獲取所有數(shù)據(jù)到內(nèi)存,不能指定多對多關(guān)系的外鍵
res = models.Book.objects.select_related('publish')prefetch_related:指定外鍵,先查詢本表所有數(shù)據(jù),再根據(jù)外鍵查詢對應(yīng)的數(shù)據(jù)表所有數(shù)據(jù),相當(dāng)于子查詢,可以指定多個(gè)外鍵,即多個(gè)子查詢
res = models.Book.objects.prefetch_related('publish')到此這篇關(guān)于Django ORM 事務(wù)和查詢優(yōu)化的文章就介紹到這了,更多相關(guān)Django ORM 事務(wù)和查詢優(yōu)化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python使用正則表達(dá)式獲取網(wǎng)頁中所需要的信息
這篇文章主要介紹了Python使用正則獲取網(wǎng)頁中所需要的信息的相關(guān)資料,需要的朋友可以參考下2018-01-01
如何利用Python實(shí)現(xiàn)簡易的音頻播放器
這篇文章主要介紹了如何利用Python實(shí)現(xiàn)簡易的音頻播放器,需要用到的庫有pygame和tkinter,實(shí)現(xiàn)音頻播放的功能,供大家學(xué)習(xí)參考,希望對你有所幫助2022-03-03
Python實(shí)現(xiàn)隨機(jī)密碼生成器實(shí)例
這篇文章主要介紹了Python實(shí)現(xiàn)隨機(jī)密碼生成器實(shí)例,string.printable是string中的可打印字符,用strip函數(shù)首尾去掉空格,random模塊用來取字符,random.choice隨機(jī)取字符,將隨機(jī)取出的字符與password空字符串進(jìn)行拼接,最后用print輸出,需要的朋友可以參考下2023-09-09
Django項(xiàng)目定期自動(dòng)清除過期session的2種方法實(shí)例
如果用戶主動(dòng)退出,session會自動(dòng)清除,如果沒有退出就一直保留,記錄數(shù)越來越大,要定時(shí)清理沒用的session,下面這篇文章主要給大家介紹了關(guān)于Django項(xiàng)目定期自動(dòng)清除過期session的2種方法,需要的朋友可以參考下2022-08-08
Python連接數(shù)據(jù)庫使用matplotlib畫柱形圖
這篇文章主要介紹了Python連接數(shù)據(jù)庫使用matplotlib畫柱形圖,文章通過實(shí)例展開對主題的相關(guān)介紹。具有一定的知識參考價(jià)值性,感興趣的小伙伴可以參考一下2022-06-06
Python3按一定數(shù)據(jù)位數(shù)格式處理bin文件的方法
今天小編就為大家分享一篇Python3按一定數(shù)據(jù)位數(shù)格式處理bin文件的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-01-01
python3中@dataclass的實(shí)現(xiàn)示例
@dataclass?是 Python 3.7 引入的一個(gè)裝飾器,用于方便地定義符合數(shù)據(jù)類協(xié)議的類,本文主要介紹了python3中@dataclass的實(shí)現(xiàn)示例,感興趣的可以了解一下2024-02-02
Python中使用kitti數(shù)據(jù)集實(shí)現(xiàn)自動(dòng)駕駛(繪制出所有物體的行駛軌跡)
這篇文章主要介紹了Python中使用kitti數(shù)據(jù)集實(shí)現(xiàn)自動(dòng)駕駛——繪制出所有物體的行駛軌跡,本次內(nèi)容主要是畫出kitti車的行駛的軌跡,需要的朋友可以參考下2022-06-06
Python根據(jù)URL地址下載文件并保存至對應(yīng)目錄的實(shí)現(xiàn)
這篇文章主要介紹了Python根據(jù)URL地址下載文件并保存至對應(yīng)目錄的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11

