django:django.utils.functional.SimpleLazyObject的目的?

don*_*ood 44 django django-views

我遇到了一个问题,我分配request.user了一个名为的变量prior_user,然后基本上验证了用户,然后检查是否request.user != prior_user.我希望它们不一样,并且prior_user应该包含`AnonymousUser.令我惊讶的是,他们是一样的.

示例代码:

prior_user = request.user   # request object, obtained froma  view
authenticate_user(request)   # some function that authenticates
print prior_user.username != request.user.username   # returns False i.e.they are the same!
Run Code Online (Sandbox Code Playgroud)

然后我发现prior_user实际上包含一个django.utils.functional.SimpleLazyObject的实例,所以我假设它是某种懒惰的查找类型的东西,即在实际使用之前不会查找prior_user的值.看一下源代码,我无法证实这一点.

有django经验的人可以告诉我发生了什么以及为什么需要它?

这让我有点动摇,因为通常的赋值语句不能按我预期的方式工作,Django中的其他内容是什么样的呢?我也没有在文档中看到这一点.

那么对django有超人类知识的人可以提供一些清晰度吗?

Chr*_*att 103

auth中间件添加user到属性request就是实例SimpleLazyObject.SimpleLazyObject,本身是一个子类LazyObject.LazyObject是,如实际代码所述:

另一个类的包装器,可用于延迟包装类的实例化

SimpleLazyObject仅通过传入方法设置该类(_wrapped属性on LazyObject),在本例中get_user.这是该方法的代码:

def get_user(request):
    if not hasattr(request, '_cached_user'):
        request._cached_user = auth.get_user(request)
    return request._cached_user
Run Code Online (Sandbox Code Playgroud)

这本身就是一个包装器auth.get_user,它实现了一种缓存机制.所以这是最终运行的:

def get_user(request):
    from django.contrib.auth.models import AnonymousUser
    try:
        user_id = request.session[SESSION_KEY]
        backend_path = request.session[BACKEND_SESSION_KEY]
        backend = load_backend(backend_path)
        user = backend.get_user(user_id) or AnonymousUser()
    except KeyError:
        user = AnonymousUser()
    return user
Run Code Online (Sandbox Code Playgroud)

所以,这里真正发生的一切request.user都是模棱两可的,直到它实际用于某事.这很重要,因为它允许它根据当前的身份验证状态进行调整.如果在进行身份验证之前访问其上的属性,它将返回一个实例AnonymousUser,但如果您进行身份验证然后访问它,则会返回一个实例User.

  • 谢谢,这对像我这样的django初学者非常有帮助! (3认同)