decorator()得到了一个意外的关键字参数

时间:2016-12-29 02:53:25

标签: python django url-pattern

我在Django视图上有这个错误:

TypeError at /web/host/1/
decorator() got an unexpected keyword argument 'host_id'
Request Method: GET
Request URL:    http://127.0.0.1:8000/web/host/1/edit
Django Version: 1.10.4
Exception Type: TypeError
Exception Value:    
decorator() got an unexpected keyword argument 'host_id'

和urlpatterns是:

 url(r'^host/(?P<host_id>[0-9]+)$', host, name='host'),

视图功能是:

@check_login
def host(request, host_id, *args, **kwargs):
    h = Host()
    # resultHost = h.get_host(host_id)
    return render(request, 'web/host.html')

下面的check_login:

def check_login(f):
    """verify if user login"""
    def decorator(request):
        if request.session.get('user', None):
            return f(request)
        else:
            return HttpResponseRedirect(reverse("web:login"))
    return decorator

如果我使用不带参数“host_id”的url和没有host_id的主机函数,程序将运行完美。

那么问题是什么?谢谢。

1 个答案:

答案 0 :(得分:4)

问题出在check_login装饰器代码中。具体问题在于:

def check_login(f):
    """verify if user login"""
    def decorator(request):  # <-- Only allows for a keyword value of 'request'
        if request.session.get('user', None):
            return f(request)
        else:
            return HttpResponseRedirect(reverse("web:login"))
    return decorator

要解决此问题,您需要接受可能传递给调用的装饰器的任何额外关键字参数。您可以通过使用可变参数来执行此操作,该参数有效地说“获取任何额外的关键字参数并将它们表示为单个值”。按照惯例,此单个值(下例中的**kwargs)是一个字典,其中键是参数的名称,值是参数值。名称kwargs是Python中常用于可变参数的约定,但不是必需的 - 您可以使用任何有效的变量名。

def check_login(f):
    """verify if user login"""
    def decorator(request, **kwargs):  # <-- **kwargs will absorb any additional keyword arguments that are passed during invocation
        if request.session.get('user', None):
            return f(request, **kwargs)
        else:
            return HttpResponseRedirect(reverse("web:login"))
    return decorator

或者为了使其更加通用,您可以接受可变参数位置关键字参数,如下所示:

def check_login(f):
    """verify if user login"""
    def decorator(request, *args, **kwargs):  # <-- *args will absorb any additional positional arguments
                                              # <-- **kwargs will absorb any additional keyword arguments
        if request.session.get('user', None):
            return f(request, *args, **kwargs)
        else:
            return HttpResponseRedirect(reverse("web:login"))
    return decorator

有关使用*args**kwargs的更多信息,我建议您在此处查看教程:https://pythontips.com/2013/08/04/args-and-kwargs-in-python-explained/