我很好奇这段代码实际上是如何工作的。特别是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添加为对象与实例的属性是一个好主意?

最佳答案

在这种情况下,由于LazyUser()是一个类,用于确定请求对象是否已检查以确保已查询用户,因此这是真正的魔力,因此每个会话只需向数据库查询一次用户。作为属性,它是一种通用操作,将应用于请求的所有实例。所述对象的所有实例的属性都相同。这只是使用户属性成为LazyUser的实例,从而仅在内存中允许它的一个实例(iirc这部分,其他人应该能够阐明这一点)。

10-06 06:22