1. 程式人生 > >Django之ORM優化查詢的方式

Django之ORM優化查詢的方式

 ORM優化查詢的方式

一、假設有三張表

Room
id 1 2 .. 1000 User: id 1 .. 10000 Booking: user_id room_id time_id date 1 1 8:00 2017-11-11 1 2 8:00 2017-11-11 1 3 8:00 2017-11-11 1 4 8:00 2017-11-11 1 5 8:00 2017-11-11

二、 需求:獲取2018-11-11所有預定資訊:

列印:使用者名稱稱,會議室名稱, 預定時間段

# 解決方案一:執行11次sql語句    
bk = models.Booking.objects.filter(date=2018-11-11)
for item in bk:
    print(item.time_id, item.room.caption, item.user.user)
    
    
# 解決方案二:執行1次
#select * from ... left join user ... join room
bk = models.Booking.objects.filter(date=2018-11-11).select_related('user','room')
for item in bk:
    print(item.time_id, item.room.caption, item.user.user)
    
# 解決方案三:執行3次
#select * from booking where date=2018-11-11
#select * from user where id in [1,]
#select * from room where id in [1,2,3,4,5]
bk = models.Booking.objects.filter(date=2018-11-11).prefetch_related('user','room')    
for item in bk:
    print(item.time_id, item.room.caption, item.user.user)

 總結:以後對於SQL語句的優化要加上selsect_releated或者prefetch_releated,這只是對於跨表做的優化,如果是單表的話就沒有必要進行優化查詢了

那麼什麼時候用selsect_releated,什麼時候用prefetch_releated呢?這個按情況而定,

selsect_releated是主動連表,執行一次SQL

prefetch_releated不連表執行3次SQL

 

 

    1.     [{} ]
        all_users = models.User.objects.all().values('
name','age','role__name') 2. [ 物件 ] all_users = models.User.objects.all() 用的時候注意,只拿自己表中的欄位,別跨表 3. select_related (外來鍵、一對一) all_users = models.User.objects.all().select_related('role') 4. prefetch_related (role) all_users = models.User.objects.all().prefetch_related('role') 5. only all_users = models.User.objects.all().only('name') 用的時候注意,只拿自己指定的欄位 6. defer all_users = models.User.objects.all().defer('name')

 

 

 

二、Q查詢的第二種方式

        remove_booking = Q()
            for room_id, time_id_list in booking_info['del'].items():
                for time_id in time_id_list:
                    temp = Q()    #例項化一個Q物件
                    temp.connector = 'AND'  #以and的方式連線
                    # user_id是一個欄位,後面的是一個欄位對應的值
                    temp.children.append(('user_id', request.session['user_info']['id'],))   
                    temp.children.append(('booking_date', booking_date,))
                    temp.children.append(('room_id', room_id,))
                    temp.children.append(('booking_time', time_id,))
                    remove_booking.add(temp, 'OR')   #以or的方式新增到temp

一、假設有三張表

Room
id 1 2 .. 1000 User: id 1 .. 10000 Booking: user_id room_id time_id date 1 1 8:00 2017-11-11 1 2 8:00 2017-11-11 1 3 8:00 2017-11-11 1 4 8:00 2017-11-11 1 5 8:00 2017-11-11

二、 需求:獲取2018-11-11所有預定資訊:

列印:使用者名稱稱,會議室名稱, 預定時間段

# 解決方案一:執行11次sql語句    
bk = models.Booking.objects.filter(date=2018-11-11)
for item in bk:
    print(item.time_id, item.room.caption, item.user.user)
    
    
# 解決方案二:執行1次
#select * from ... left join user ... join room
bk = models.Booking.objects.filter(date=2018-11-11).select_related('user','room')
for item in bk:
    print(item.time_id, item.room.caption, item.user.user)
    
# 解決方案三:執行3次
#select * from booking where date=2018-11-11
#select * from user where id in [1,]
#select * from room where id in [1,2,3,4,5]
bk = models.Booking.objects.filter(date=2018-11-11).prefetch_related('user','room')    
for item in bk:
    print(item.time_id, item.room.caption, item.user.user)

 總結:以後對於SQL語句的優化要加上selsect_releated或者prefetch_releated,這只是對於跨表做的優化,如果是單表的話就沒有必要進行優化查詢了

那麼什麼時候用selsect_releated,什麼時候用prefetch_releated呢?這個按情況而定,

selsect_releated是主動連表,執行一次SQL

prefetch_releated不連表執行3次SQL

 

 

    1.     [{} ]
        all_users = models.User.objects.all().values('name','age','role__name')
        
    2. [ 物件 ]
        all_users = models.User.objects.all()
        用的時候注意,只拿自己表中的欄位,別跨表
    3. select_related  (外來鍵、一對一)
        all_users = models.User.objects.all().select_related('role')
        
    4. prefetch_related (role)
        all_users = models.User.objects.all().prefetch_related('role')
        
    5. only
        all_users = models.User.objects.all().only('name')
        用的時候注意,只拿自己指定的欄位
    6. defer
        all_users = models.User.objects.all().defer('name')

 

 

 

二、Q查詢的第二種方式

        remove_booking = Q()
            for room_id, time_id_list in booking_info['del'].items():
                for time_id in time_id_list:
                    temp = Q()    #例項化一個Q物件
                    temp.connector = 'AND'  #以and的方式連線
                    # user_id是一個欄位,後面的是一個欄位對應的值
                    temp.children.append(('user_id', request.session['user_info']['id'],))   
                    temp.children.append(('booking_date', booking_date,))
                    temp.children.append(('room_id', room_id,))
                    temp.children.append(('booking_time', time_id,))
                    remove_booking.add(temp, 'OR')   #以or的方式新增到temp