Python在类下定义函数要用self参数是因为类的方法与普通的函数特别的区别是必须有一个额外的名列前茅个参数名称,但是在调用这个方法时不需为这个参数赋值,Python会提供这个值。这个特别的变量指对象本身,按照惯例它的名称是self。
一、Python在类下定义函数要用self参数
Python在类下定义函数要用self参数是因为类的方法与普通的函数特别的区别是必须有一个额外的名列前茅个参数名称,但是在调用这个方法时不需为这个参数赋值,Python会提供这个值。这个特别的变量指对象本身,按照惯例它的名称是self。
虽然你可以给这个参数任何名称,但是 强烈建议 你使用self这个名称——其他名称都是不赞成你使用的。使用一个标准的名称有很多优点——你的程序读者可以迅速识别它,如果使用self的话,还有些IDE(集成开发环境)也可以帮助你。
Python中的self等价于C++中的self指针和Java、C#中的this参考。
你一定很奇怪Python如何给self赋值以及为何你不需要给它赋值。举一个例子会使此变得清晰。假如你有一个类称为MyClass和这个类的一个实例MyObject。当你调用这个对象的方法MyObject.method(arg1, arg2)的时候,这会由Python自动转为MyClass.method(MyObject, arg1, arg2)——这就是self的原理了。
这也意味着如果你有一个不需要参数的方法,你还是得给这个方法定义一个self参数。
我这里补充一下几个实例程序以加深理解:
PHP
1 2 3 4 5 | class Person: def sayHi(self): print ‘Hello,how are you?’ p=Person() p.sayHi() |
结果显示为:
Hello,how are you?
改一下
PHP
1 2 3 4 5 | class Person: def sayHi(self): print ‘Hello,how are you?’ p=Person() Person.sayHi(p) |
结是同上!
再改一下:(去掉self,会怎么样呢?)
PHP
1 2 3 4 5 6 | class Person: def sayHi(): print ‘Hello,how are you?’ p=Person() Person.sayHi(p) p.sayHi() |
出现如下错误信息:
Traceback (most recent call last):
File “C:\pro\Person2.py”, line 5, in ?
Person.sayHi(p)
TypeError: sayHi() takes no arguments (1 given)
__name__是用来识别一个模块是直接运行还是作为一般的模块被导入的状态。当一个模块是直接运行时,__name__就等于__main__,如果它是作为一般模块被导入时,__name__就是模块本身的名字。
示例:
PHP
1 2 | if __name__ == “__main__”: run() |
self是一种约定。在Python中,类方法的名列前茅个参数表示对象本身,在Python中一般使用self。你也可以使用this.
示例:
a是你想用的方法。但self是一种约定。在调用一个对象的方法时,对象本身被作为self参数传入。如:
PHP
1 2 3 4 5 | class A: def p(self, name): print name a=A() a.p(‘aa’) |
这时,相当于A.p(a, ‘aa’)
Python的对象不一定要有__init__方法,无论它是基类还是子类。因为__init__不是构造函数,Python对象在调用__init__的时候已被构造出来,称为“初始化”比较合适。
如果一个对象是子类,且需要有__init__,那么在__init__的名列前茅个语句就是父类的__init__(除非父类是像dict一样的内置数据类型则不需要__init__)。调用完后再进行子类的相关初始化,作为方法覆盖。
一般在子类覆盖“专有类方法”,并触发相关动作,这样不必使用__init__也能进行更高级的初始化。
如果子类的方法覆盖了父类,想调用父类的方法怎么吗?Python没有super啊。很简单,直接“父类名.方法”即可。
一般在类定义中方法之外的属性叫“类属性”,它跟定义在方法内的属性有什么区别呢?“ 类属性”可以使得不必生成实例也可以调用,而嵌在方法内的(特别是由__init__初始化后产生的属性),一定得产生实例才可以调用之。
所有的类的实例都享有统一的“类属性”,那这“类属性”是否就是类的常量?不是!通过在任意实例调用self.__class__.NAME就可以修改“类属性”,修改后所有的实例都将应用这一新的“类属性”。self.__class__即代表类的引用,不代表任何实例,实例就是self。
延伸阅读:
二、垃圾回收机制
当内存中有不再使用的部分时,垃圾收集器就会把他们清理掉。它会去检查那些引用计数为0的对象,然后清除其在内存的空间。当然除了引用计数为0的会被清除,还有一种情况也会被垃圾收集器清掉:当两个对象相互引用时,他们本身其他的引用已经为0了。
垃圾回收机制还有一个循环垃圾回收器, 确保释放循环引用对象(a引用b, b引用a, 导致其引用计数永远不为0)。