我很好奇这段代码是如何运作的。特别是request.__class__.user = LazyUser()
与setattr(request, user, LazyUser())
相反的做法。很抱歉,如果这是一个无知的问题,我想知道,因为我将实现一个类似的中间件,为请求添加一个属性。
以下是代码,当然,当您调用request.user
时,您会返回某种用户对象。
class LazyUser(object):
def __get__(self, request, obj_type=None):
if not hasattr(request, '_cached_user'):
from django.contrib.auth import get_user
request._cached_user = get_user(request)
return request._cached_user
class AuthenticationMiddleware(object):
def process_request(self, request):
assert hasattr(request, 'session'), "The Django authentication middleware requires session middleware to be installed. Edit your MIDDLEWARE_CLASSES setting to insert 'django.contrib.sessions.middleware.SessionMiddleware'."
request.__class__.user = LazyUser()
return None
更新:做一点阅读,我理解__get__
位是如何工作的,所以我的问题归结为为什么这样做与使用setattr()
如上所述。为什么将user
作为对象的属性添加到实例是一个好主意?
答案 0 :(得分:3)
在这种情况下,由于LazyUser()是一个类,它确定请求对象是否已经检查以确保已查询用户,这是真正的魔力,它只需要为用户查询数据库每次一次。作为属性,它是一个通用操作,将应用于请求的所有实例。所述对象的所有实例的属性都相同。这简单地使用户属性成为LazyUser的一个实例,只允许它在内存中的一个实例(iirc这一部分,其他人应该能够澄清这一点)。
答案 1 :(得分:0)
为什么将用户作为对象的属性添加到实例是一个好主意?
这是必要的,所以__get__方法是通过描述符协议调用的。请参阅here。