“TypeError:method() 接受 1 个位置参数,但给出了 2 个”,但我只传递了一个

"TypeError: method() takes 1 positional argument but 2 were given" but I only passed one

提问人:Zero Piraeus 提问时间:5/30/2014 最后编辑:wjandreaZero Piraeus 更新时间:9/22/2023 访问量:1373580

问:

如果我有一个班级......

class MyClass:

    def method(arg):
        print(arg)

...我用它来创建一个对象......

my_object = MyClass()

...我这样称呼......method("foo")

>>> my_object.method("foo")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: method() takes exactly 1 positional argument (2 given)

...为什么 Python 告诉我我给了它两个参数,而我只给了一个?

python 方法 参数 self

评论

35赞 smci 5/2/2020
这个信息有无数的原因;这里的具体原因是所有实例方法都期望第一个 Arg,根据惯例,我们称之为 。所以声明对于一种方法来说是错误的,它应该是 .当方法调度尝试调用并匹配两个参数时,您会收到该错误。selfdef method(arg):def method(self, arg):method(arg):self, arg

答:

623赞 Zero Piraeus 5/30/2014 #1

在 Python 中,这个:

my_object.method("foo")

...是句法糖,口译员在幕后将其翻译为:

MyClass.method(my_object, "foo")

...正如你所看到的,它确实有两个参数 - 只是从调用者的角度来看,第一个是隐含的。

这是因为大多数方法都会对它们所调用的对象进行一些工作,因此需要有某种方式在方法中引用该对象。按照惯例,第一个参数在方法定义中调用:self

class MyNewClass:

    def method(self, arg):
        print(self)
        print(arg)

如果调用 的实例,则按预期工作:method("foo")MyNewClass

>>> my_new_object = MyNewClass()
>>> my_new_object.method("foo")
<__main__.MyNewClass object at 0x29045d0>
foo

偶尔(但不经常),你真的关心你的方法绑定到的对象,在这种情况下,你可以用内置的 staticmethod() 函数来装饰方法,这样说:

class MyOtherClass:

    @staticmethod
    def method(arg):
        print(arg)

...在这种情况下,您不需要向方法定义添加参数,它仍然有效:self

>>> my_other_object = MyOtherClass()
>>> my_other_object.method("foo")
foo

评论

186赞 amoebe 6/10/2017
简而言之:将第一个参数添加到方法中可以解决问题。self
5赞 Ujjwal 9/6/2020
或者只是在定义之前添加@staticmethod。凉
0赞 joeschmoe54321 6/21/2023
你能在这里详细说明一下这一点吗:“偶尔(但不经常),你真的不关心你的方法绑定到的对象......”我有 Java 和 .NET 背景,我经常将方法组织到适当的类中进行分组/组织。这在 Python 中也无效吗?
2赞 Zero Piraeus 6/22/2023
@joeschmoe54321在 Python 中,您通常使用模块和包来对函数进行分组,而不是类。
25赞 Jonru 1/4/2016 #2

遇到此类错误时要考虑的其他事项:

我遇到了这个错误消息,发现这篇文章很有帮助。事实证明,在我的情况下,我覆盖了存在对象继承的地方。__init__()

继承的示例相当长,因此我将跳到一个不使用继承的更简单的示例:

class MyBadInitClass:
    def ___init__(self, name):
        self.name = name

    def name_foo(self, arg):
        print(self)
        print(arg)
        print("My name is", self.name)


class MyNewClass:
    def new_foo(self, arg):
        print(self)
        print(arg)


my_new_object = MyNewClass()
my_new_object.new_foo("NewFoo")
my_bad_init_object = MyBadInitClass(name="Test Name")
my_bad_init_object.name_foo("name foo")

结果是:

<__main__.MyNewClass object at 0x033C48D0>
NewFoo
Traceback (most recent call last):
  File "C:/Users/Orange/PycharmProjects/Chapter9/bad_init_example.py", line 41, in <module>
    my_bad_init_object = MyBadInitClass(name="Test Name")
TypeError: object() takes no parameters

PyCharm 没有发现这个错别字。Notepad++(其他编辑器/IDE的可能)也没有。

当然,这是一个“不带参数”的 TypeError,就 Python 中的对象初始化而言,它与期望一个时的“得到两个”没有太大区别。

解决该主题:如果语法正确,将使用重载初始值设定项,但如果不是,它将被忽略,而是使用内置的初始值设定项。该对象不会期望/处理此问题,并引发错误。

如果出现 sytax 错误: 修复很简单,只需编辑自定义 init 语句:

def __init__(self, name):
    self.name = name

评论

3赞 abarnert 6/28/2018
这里没有。代码在语法上是正确的;它只是正确地定义了一个名为 的方法,没有人会调用它,而不是特殊方法。这就是未检测到错误的原因,因为没有可检测的错误。SyntaxError___init____init__
0赞 anonymous 9/6/2022
完全相同的事情发生在我身上,但有.由于这个错别字而浪费了几个小时。__int__(self, ...):
0赞 wjandrea 11/16/2022
我很高兴你找到了解决方案,但这是一个与问题中的问题完全不同的问题。我想说你可以发布你自己的问题,但同样的事情已经发布过了,例如:“这个构造函数没有参数”__init__错误(错别字是)。_init_
9赞 Trinadh Koya 2/3/2017 #3

当您未指定参数的编号或任何其他方法查找时,就会发生这种情况。__init__()

例如:

class Dog:
    def __init__(self):
        print("IN INIT METHOD")

    def __unicode__(self,):
        print("IN UNICODE METHOD")

    def __str__(self):
        print("IN STR METHOD")

obj = Dog("JIMMY", 1, 2, 3, "WOOF")

当您运行上述程序时,它会给出如下错误:

TypeError: __init__() takes 1 positional argument but 6 were given

我们怎样才能摆脱这个东西?

只要传递参数,找什么方法__init__()

class Dog:
    def __init__(self, dogname, dob_d, dob_m, dob_y, dogSpeakText):
        self.name_of_dog = dogname
        self.date_of_birth = dob_d
        self.month_of_birth = dob_m
        self.year_of_birth = dob_y
        self.sound_it_make = dogSpeakText

    def __unicode__(self, ):
        print("IN UNICODE METHOD")

    def __str__(self):
        print("IN STR METHOD")


obj = Dog("JIMMY", 1, 2, 3, "WOOF")
print(id(obj))
47赞 simhumileco 6/19/2019 #4

简单来说

在 Python 中,您应该将第一个参数添加到类中所有定义的方法中:self

class MyClass:
  def method(self, arg):
    print(arg)

然后你可以根据你的直觉使用你的方法:

>>> my_object = MyClass()
>>> my_object.method("foo")
foo

为了更好地理解,您还可以阅读这个问题的答案:自我的目的是什么?

评论

4赞 simhumileco 8/9/2019
这个答案有什么问题?为什么有人给她一个负面的分数?毕竟,它是问题的答案,与其他答案相比,它的特点是简单,这对于一些正在寻找答案的人来说可能很重要。不是吗?
3赞 grafuls 5/27/2022
说你应该在类中向所有定义的方法添加参数是错误的。在上面的示例中,方法内部没有用处,因此可以对其进行修饰,从而不需要创建类的实例。你可以在这里阅读更多关于它的信息:docs.python.org/3/library/......selfself@staticmethod
2赞 wjandrea 11/16/2022
也可以有 @classmethods,它们作为它们的第一个参数,而不是 .clsself
21赞 Stanislav Pankevich 7/28/2019 #5

此问题也可能是由于未能正确将关键字参数传递给函数而导致的。

例如,给定一个定义如下的方法:

def create_properties_frame(self, parent, **kwargs):

像这样的电话:

self.create_properties_frame(frame, kw_gsp)

将导致 ,因为字典被视为位置参数,而不是被解压缩到单独的关键字参数中。TypeError: create_properties_frame() takes 2 positional arguments but 3 were givenkw_gsp

解决方案是添加到参数中:**

self.create_properties_frame(frame, **kw_gsp)

评论

1赞 Little Brain 8/23/2019
来自 JavaScript,我希望能够传递列表或字典作为参数。但似乎这不是 Python 的工作方式:你必须用 * 或 ** 来解构列表,后者用于关键字参数列表。
3赞 smci 5/2/2020
@LittleBrain:是的,如果您希望它们在函数中被视为列表(/dict),您可以传递一个列表(或字典)。但是,如果您希望它们的各个值解压缩并与函数中的参数 (/kwargs) 匹配,则需要 (/) 语法。*args**kwargs
-1赞 Coddy 1/18/2020 #6

您实际上应该创建一个类:

class accum:
    def __init__(self):
        self.acc = 0
    def accumulator(self, var2add, end):
        if not end:
            self.acc+=var2add
        return self.acc

评论

1赞 wjandrea 11/16/2022
这如何回答这个问题?
-1赞 Gabriel Arghire 6/6/2020 #7

就我而言,我忘了添加()

我是这样称呼这个方法的

obj = className.myMethod

但应该是这样的

obj = className.myMethod()

评论

0赞 wjandrea 11/16/2022
我不关注。这将导致问题中的错误吗?
0赞 Gabriel Arghire 11/18/2022
它只是奏效了。
15赞 Rotem jackoby 7/2/2020 #8

正如其他答案中提到的 - 当您使用实例方法时,您需要作为第一个参数传递 - 这是错误的根源。self

除此之外,重要的是要了解只有实例方法将 self 作为第一个参数才能引用实例

如果方法是静态的,则不传递 ,而是传递一个参数(或)。selfclsclass_

请看下面的例子。

class City:

   country = "USA" # This is a class level attribute which will be shared across all instances  (and not created PER instance)

   def __init__(self, name, location, population):
       self.name       = name
       self.location   = location
       self.population = population
 
   # This is an instance method which takes self as the first argument to refer to the instance 
   def print_population(self, some_nice_sentence_prefix):
       print(some_nice_sentence_prefix +" In " +self.name + " lives " +self.population + " people!")

   # This is a static (class) method which is marked with the @classmethod attribute
   # All class methods must take a class argument as first param. The convention is to name is "cls" but class_ is also ok
   @classmethod
   def change_country(cls, new_country):
       cls.country = new_country

一些测试只是为了让事情更清楚:

# Populate objects
city1 = City("New York",    "East", "18,804,000")
city2 = City("Los Angeles", "West", "10,118,800")

#1) Use the instance method: No need to pass "self" - it is passed as the city1 instance
city1.print_population("Did You Know?") # Prints: Did You Know? In New York lives 18,804,000 people!

#2.A) Use the static method in the object
city2.change_country("Canada")

#2.B) Will be reflected in all objects
print("city1.country=",city1.country) # Prints Canada
print("city2.country=",city2.country) # Prints Canada

评论

1赞 wjandrea 11/16/2022
这个术语是错误的。classmethodsstaticmethods 是两个不同的东西。s take 而 s 没有任何必需的参数。classmethodclsstaticmethod
0赞 wjandrea 11/16/2022
另外,我不认为这是一个很好的例子,因为你为什么要让一个实例影响所有其他实例?比如,为什么这个类一开始就有一个属性?每个实例不应该定义自己的国家吗?假设我想为东京做一个实例,为多伦多再做一个实例;这是不可能的,那么它有什么用呢?下面是一个更好的示例,包括 .classmethodCityCitycountryCityCitystaticmethod
0赞 Rotem jackoby 11/16/2022
谢谢@wjandrea,我不是 100% 确定我理解了您的评论以及为什么我的答案以任何方式提供了错误的信息。
5赞 Simon Alford 3/9/2021 #9

当我睡眠不足时,我收到此错误,并使用以下方法创建一个类:defclass

def MyClass():
    def __init__(self, x):
        self.x = x

a = MyClass(3)
-> TypeError: MyClass() takes 0 positional arguments but 1 was given

评论

0赞 wjandrea 11/16/2022
我很高兴你找到了解决方案,但这是一个与问题中的问题完全不同的问题。我想说你可以发布你自己的问题,但同样的事情已经发布:TypeError:__init__ 需要 1 个位置参数,但给出了 2 个。
1赞 run_the_race 1/28/2023
@wjandrea,这个答案是针对主题的确切错误消息。我不了解你,但我不读问题,我读主题和答案。
0赞 wjandrea 1/28/2023
@run_the_race是的,最好阅读整个问题,但我有时自己会跳过它。因此,我更新了标题以更好地反映问题。
6赞 Manikandan Raju 8/12/2021 #10

如果要在不创建对象的情况下调用方法,可以将 method 更改为 static method。

class MyClass:

    @staticmethod
    def method(arg):
        print(arg)

MyClass.method("i am a static method")
0赞 King Chudi 3/30/2023 #11

如果你在 Django 中遇到过这种情况,那么这就是它的含义:

  1. 在函数中添加一个对象,Django 会理解其余的,示例
def display_genre(self, obj):
        return ', '.join(genre.name for genre in obj.genre.all())}

0赞 cottontail 9/22/2023 #12

如果函数采用一个位置参数和任何其他变量作为关键字参数,您也可能会收到相同的错误消息。若要解决此问题,请将其他变量作为关键字参数传递。

触发错误的示例:

def func(value, **kwargs):
    pass

func(1, 2)   # <--- TypeError: func() takes 1 positional argument but 2 were given

func(1, b=2) # <--- OK

另一个更实际的例子:django 的方法可用于验证用户名和密码等凭据,但它将它们作为关键字参数(文档源代码),因此会出现以下场景:authenticate()

from django.contrib.auth import authenticate
authenticate(username, password)                    # <--- TypeError
authenticate(username=username, password=password)  # <--- OK