【问题标题】:__init__() got an unexpected keyword argument 'user'__init__() 得到了一个意外的关键字参数“用户”
【发布时间】:2013-11-27 23:04:22
【问题描述】:

我在创建用户时使用 Django 创建用户和对象。但是有一个错误

__init__() got an unexpected keyword argument 'user'

在 view.py 中调用 register() 函数时。 功能是:

def register(request):  
    '''signup view'''     
    if request.method=="POST":  
        form=RegisterForm(request.POST)  
        if form.is_valid():  
            username=form.cleaned_data["username"]  
            email=form.cleaned_data["email"]  
            password=form.cleaned_data["password"]  
            user=User.objects.create_user(username, email, password)  
            user.save()
            return HttpResponseRedirect('/keenhome/accounts/login/')
        else: 
            form = RegisterForm()      
            return render_to_response("polls/register.html", {'form':form}, context_instance=RequestContext(request))  

    #This is used for reinputting if failed to register    
    else: 
        form = RegisterForm()      
        return render_to_response("polls/register.html", {'form':form}, context_instance=RequestContext(request))

对象类是:

class LivingRoom(models.Model):
    '''Living Room object'''
    user = models.OneToOneField(User)

    def __init__(self, temp=65):
        self.temp=temp

    TURN_ON_OFF = (
        ('ON', 'On'),
        ('OFF', 'Off'),
    )

    TEMP = (
        ('HIGH', 'High'),
        ('MEDIUM', 'Medium'),
        ('LOW', 'Low'),
    )

    on_off = models.CharField(max_length=2, choices=TURN_ON_OFF)
    temp = models.CharField(max_length=2, choices=TEMP)

#signal function: if a user is created, add control livingroom to the user    
def create_control_livingroom(sender, instance, created, **kwargs):
    if created:
        LivingRoom.objects.create(user=instance)

post_save.connect(create_control_livingroom, sender=User)

Django错误页面通知错误信息: user=User.objects.create_user(username, email, password)LivingRoom.objects.create(user=instance)

我试图搜索这个问题,找到了一些案例,但仍然无法弄清楚如何解决。

【问题讨论】:

  • 我的猜测是,这里失败了:create_control_livingroom 你能关掉信号试试吗?你也可以显示堆栈跟踪吗?
  • 那你为什么把LivingRoom.__init__方法限制为只是 temp呢?

标签: python django django-models django-forms django-views


【解决方案1】:

你做不到

LivingRoom.objects.create(user=instance)

因为您有一个不将 user 作为参数的 __init__ 方法。

你需要类似的东西

#signal function: if a user is created, add control livingroom to the user    
def create_control_livingroom(sender, instance, created, **kwargs):
    if created:
        my_room = LivingRoom()
        my_room.user = instance

更新

但是,由于bruno 具有already said it,Django 的models.Model 子类的初始化程序最好不要管,或者应该接受与模型元字段匹配的*args**kwargs

所以,遵循更好的原则,你可能应该有类似的东西

class LivingRoom(models.Model):
    '''Living Room object'''
    user = models.OneToOneField(User)

    def __init__(self, *args, temp=65, **kwargs):
        self.temp = temp
        return super().__init__(*args, **kwargs)

注意 - 如果您没有使用 temp 作为关键字参数,例如LivingRoom(65),那么你必须开始这样做。 LivingRoom(user=instance, temp=66) 或者如果您想要默认值 (65),只需 LivingRoom(user=instance) 即可。

【讨论】:

  • 是的,问题出在__init__()函数上,但是为什么does NOT take user as argument这个函数呢?
  • 因为函数“init”只有两个可接受的变量def __init__(self, temp=65)self(类实例)和temp。此外,尚未为该实例定义类变量“user”。
【解决方案2】:

检查您的进口。可能有两个具有相同名称的类。来自您的代码或您正在使用的库。就个人而言,这就是问题所在。

【讨论】:

    【解决方案3】:

    我遇到了同样的错误。

    在我看来,我是这样重写 get_form_kwargs() 的:

    class UserAccountView(FormView):
        form_class = UserAccountForm
        success_url = '/'
        template_name = 'user_account/user-account.html'
    
    def get_form_kwargs(self):
        kwargs = super(UserAccountView, self).get_form_kwargs()
        kwargs.update({'user': self.request.user})
        return kwargs
    

    但在我的表单上,我未能覆盖 init() 方法。一旦我做到了。问题解决了

    class UserAccountForm(forms.Form):
        first_name = forms.CharField(label='Your first name', max_length=30)
        last_name = forms.CharField(label='Your last name', max_length=30)
        email = forms.EmailField(max_length=75)
    
        def __init__(self, *args, **kwargs):
            user = kwargs.pop('user')
            super(UserAccountForm, self).__init__(*args, **kwargs)
    

    【讨论】:

    • 只是提供一个注释。我很确定如果你在user = kwargs.pop('user') 之前调用super(UserAccountForm, self).__init__(*args, **kwargs),就会出错。我遇到了一些问题,最终只是开始将实际参数传递给__init__,但如果你说这可行,我觉得这就是原因。
    【解决方案4】:

    LivingRoom.objects.create() 调用LivingRoom.__init__() - 如果您已经阅读了回溯,您可能已经注意到 - 将相同的参数传递给它。长话短说,Django models.Model 子类的初始化器最好不要管,或者应该接受匹配模型元字段的 *args 和 **kwargs。为字段提供默认值的正确方法是在字段构造函数中使用 default 关键字,如 FineManual 中所述。

    【讨论】:

      猜你喜欢
      • 2021-11-17
      • 2015-06-08
      • 1970-01-01
      • 2021-03-17
      • 2013-05-05
      • 2016-06-05
      • 2020-08-20
      • 2021-09-20
      • 2014-05-12
      相关资源
      最近更新 更多