提问人:Zero Piraeus 提问时间:5/30/2014 最后编辑:wjandreaZero Piraeus 更新时间:9/22/2023 访问量:1373580
“TypeError:method() 接受 1 个位置参数,但给出了 2 个”,但我只传递了一个
"TypeError: method() takes 1 positional argument but 2 were given" but I only passed one
问:
如果我有一个班级......
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 中,这个:
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
评论
self
遇到此类错误时要考虑的其他事项:
我遇到了这个错误消息,发现这篇文章很有帮助。事实证明,在我的情况下,我覆盖了存在对象继承的地方。__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
评论
SyntaxError
___init__
__init__
__int__(self, ...):
__init__
错误(错别字是)。_init_
当您未指定参数的编号或任何其他方法查找时,就会发生这种情况。__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))
简单来说
在 Python 中,您应该将第一个参数添加到类中所有定义的方法中:self
class MyClass:
def method(self, arg):
print(arg)
然后你可以根据你的直觉使用你的方法:
>>> my_object = MyClass()
>>> my_object.method("foo")
foo
为了更好地理解,您还可以阅读这个问题的答案:自我的目的是什么?
评论
self
self
@staticmethod
此问题也可能是由于未能正确将关键字参数传递给函数而导致的。
例如,给定一个定义如下的方法:
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 given
kw_gsp
解决方案是添加到参数中:**
self.create_properties_frame(frame, **kw_gsp)
评论
*args
**kwargs
您实际上应该创建一个类:
class accum:
def __init__(self):
self.acc = 0
def accumulator(self, var2add, end):
if not end:
self.acc+=var2add
return self.acc
评论
就我而言,我忘了添加()
我是这样称呼这个方法的
obj = className.myMethod
但应该是这样的
obj = className.myMethod()
评论
正如其他答案中提到的 - 当您使用实例方法时,您需要作为第一个参数传递 - 这是错误的根源。self
除此之外,重要的是要了解只有实例方法将 self
作为第一个参数才能引用实例。
如果方法是静态的,则不传递 ,而是传递一个参数(或)。self
cls
class_
请看下面的例子。
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
评论
classmethod
s 和 staticmethod
s 是两个不同的东西。s take 而 s 没有任何必需的参数。classmethod
cls
staticmethod
classmethod
City
City
country
City
City
staticmethod
当我睡眠不足时,我收到此错误,并使用以下方法创建一个类:def
class
def MyClass():
def __init__(self, x):
self.x = x
a = MyClass(3)
-> TypeError: MyClass() takes 0 positional arguments but 1 was given
评论
__init__
需要 1 个位置参数,但给出了 2 个。
如果要在不创建对象的情况下调用方法,可以将 method 更改为 static method。
class MyClass:
@staticmethod
def method(arg):
print(arg)
MyClass.method("i am a static method")
如果你在 Django 中遇到过这种情况,那么这就是它的含义:
- 在函数中添加一个对象,Django 会理解其余的,示例
def display_genre(self, obj):
return ', '.join(genre.name for genre in obj.genre.all())}
如果函数采用一个位置参数和任何其他变量作为关键字参数,您也可能会收到相同的错误消息。若要解决此问题,请将其他变量作为关键字参数传递。
触发错误的示例:
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
上一个:Scala 中方法和函数的区别
下一个:使用 C 将方法作为参数传递#
评论
self
def method(arg):
def method(self, arg):
method(arg):
self, arg