所以,我刚开始学习Python(使用Codecademy),我有点困惑。
为什么有一些方法需要参数,而其他方法使用点符号?
len()接受一个参数,但不能使用点表示法:
>>> len("Help")
4
>>>"help".len()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'str' object has no attribute 'len'
同样地:
>>>"help".upper()
'HELP'
>>>upper("help")
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'upper' is not defined
这里的关键词是方法。函数和方法之间存在细微差别。
是在给定对象的类中定义的函数。例如:
class Dog:
def bark(self):
print 'Woof woof!'
rufus = Dog()
rufus.bark() # called from the object
函数是全局定义的过程:
def bark():
print 'Woof woof!'
至于你关于len
函数的问题,全局定义的函数调用对象的__len__
特殊方法。所以在这种情况下,这是一个可读性问题。
否则,当它们仅应用于某些对象时,方法会更好。应用于多个对象时,函数更好。例如,你怎么能大写一个数字?您不会将其定义为函数,您只能将其定义为仅在字符串类中的方法。
你所谓的“点符号”是类方法,它们只适用于具有类实现者定义的方法的类。 len
是一个内置函数,它接受一个参数并返回该对象的大小。一个类可以实现一个名为len
的方法,如果它想要,但大多数不需要。内置的len
函数有一个规则,说如果一个类有一个名为__len__
的方法,它将使用它,所以这适用:
>>> class C(object):
... def __len__(self):
... return 100
...
>>> len(C())
100
"help".upper
是相反的。字符串类定义了一个名为upper
的方法,但这并不意味着必须有一个名为upper
的函数。事实证明在upper
模块中有一个string
函数,但通常你不必仅仅因为你实现了一个类方法就实现了一个额外的函数。
这是函数和方法之间的区别。如果您只是学习基础知识,可能只是接受这种差异存在,并且您最终会理解它。
还在?实际上,它甚至都不难。在面向对象的编程中,方法比许多事物的函数更受欢迎,因为这意味着一种类型的对象可以覆盖其方法版本而不影响系统的其余部分。
例如,让我们假装你有一种新的字符串,当你调用.upper()
时,重音字符应该失去重音。这种类型的实例可以继承str
,并且在其他方面表现完全相同,基本上是免费的;他们需要重新定义的是upper
方法(即便如此,可能会调用基类的方法,只在处理带重音的小写字符时才更改逻辑)。期望处理字符串的软件将继续工作,如果你传入一个预期标准str
的新类型的对象,它甚至不知道差异。
Python中的设计原则是一切都是对象。这意味着您甚至可以为object
,class
和type
等基本基础对象创建自己的替换,即扩展或覆盖应用程序或平台的基本语言。
事实上,当unicode
字符串被引入该语言时,这发生在Python 2中。许多应用程序软件继续像以前一样工作,但现在使用unicode
实例,其中之前编写的代码用于处理str
实例。 (这种差异在Python 3中不再存在;或者更确切地说,被称为str
并且几乎在所有地方使用的类型现在称为bytes
,仅在您特别想要处理非文本数据时使用。)
回到我们新的upper
方法,考虑相反的情况;如果upper
只是标准库中的一个函数,你怎么会考虑修改需要upper
表现不同的软件?如果明天你的老板要你为lower
做同样的事怎么办?这将是一项艰巨的任务,你必须在整个代码库中进行的更改很容易倾向于意大利面结构,并且可能会引入微妙的新错误。
这是面向对象编程的基石之一,但是当你在更有条理的介绍中学习其他两三个原则时,它可能才真正有用。就目前而言,也许快速而肮脏的总结是“方法使实现模块化和可扩展”。