相关概念区别说明:
1. self表示为类型为类的object,而cls表示为类也就是class
2. 在外部调用静态方法时,可以使用"类名.方法名"的方式,也可以使用"对象名.方法名"的方式。而实例方法只有后面这种方式。也就是说,调用静态方法可以无需创建对象。
3. 静态方法参数没有实例参数 self, 也就不能调用实例参数。静态方法主要用来存放逻辑性的代码,基本在静态方法中,不会涉及到类的方法和类的参数。静态方法在访问本类的成员时,只允许访问静态成员(即静态成员变量和静态方法),而不允许访问实例成员变量和实例方法;实例方法则无此限制。从语法上讲,静态方法类似于全局函数,但是隐藏在类的定义里,语法上看起来比全局函数清晰的很多,习惯OO思维的人很容易接受静态方法这个概念。
4. 在定义类方法的时候,传递的参数为cls,表示为类,是将类本身作为对象进行操作的方法。类方法可以被对象调用,也可以被实例调用;传入的都是类对象,主要用于工厂方法,具体的实现就交给子类处理
5. 在重载调用父类方法的时候,最好是使用super来进行调用父类的方法。
1实例方法
实例方法就是类的实例能够使用的方法。如下:
class Foo:
def __init__(self, name):
self.name = name
def hi(self):
print self.name
if __name__ == '__main__':
foo01 = Foo('dave@www.cndba.cn') # 实例方法
foo01.hi()
print type(Foo)
print type(foo01)
print id(foo01)
print id(Foo)
运行结果为:
dave@www.cndba.cn
57293320
53014120
可以看到,Foo的type为classobj(类对象,python中定义的类本身也是对象),foo01的type为instance(实例)。而hi()是实例方法,所以foo01.hi()会输出' dave@www.cndba.cn '。实例方法的第一个参数默认为self,代指实例。self不是一个关键字,而是约定的写法。init()是生成实例时默认调用的实例方法。
2静态方法
静态方法是一种普通函数,就位于类定义的命名空间中,它不会对任何实例类型进行操作。使用装饰器@staticmethod定义静态方法。类对象和实例都可以调用静态方法:
class Foo:
def __init__(self, name):
self.name = name
def hi(self):
print self.name
@staticmethod
def add(a, b):
print a + b
if __name__ == '__main__':
foo01 = Foo('http://www.cndba.cn')
foo01.hi()
foo01.add(1, 2) # 实例对象
Foo.add(1, 2) # 类对象
运行结果如下:
http://www.cndba.cn
3
3
3类方法
类方法是将类本身作为对象进行操作的方法。类方法使用@classmethod装饰器定义,其第一个参数是类,约定写为cls。类对象和实例都可以调用类方法:
class Foo:
name = 'http://www.cndba.cn/dave '
@classmethod
def hi(cls, x):
print cls.name * x
if __name__ == '__main__':
foo01 = Foo()
foo01.hi(2) # 实例对象
Foo.hi(3) # 类对象
运行结果如下:
http://www.cndba.cn/dave http://www.cndba.cn/dave
http://www.cndba.cn/dave http://www.cndba.cn/dave http://www.cndba.cn/dave
4super
super用来执行父类中的函数,例如:
class Foo(object):
def hi(self):
print 'welcome to http://www.cndba.cn!'
class Foo2(Foo):
def hi(self):
super(Foo2, self).hi()
if __name__ == '__main__':
foo2 = Foo2()
foo2.hi()
运行结果:
welcome to http://www.cndba.cn!
注意,Foo类必须继承某个类(并且这个继承链开始于object类),否则会报错:TypeError: must be type, not classobj。
5类变量和实例变量
类变量定义在类的定义之后,实例变量则是以为self.开头。例如:
class Foo(object):
val = 0 # 类变量
def __init__(self):
self.val = 1 # 实例变量
if __name__ == '__main__':
foo = Foo()
print foo.val # 实例也可以访问类变量
print Foo.val
6如何调用父类的构造函数
子类(派生类)并不会自动调用父类(基类)的init方法,例如:
class Foo(object):
def __init__(self):
self.val = 1
class Foo2(Foo):
def __init__(self):
print self.val
if __name__ == '__main__':
foo2 = Foo2()
运行时报错:AttributeError: 'Foo2' object has no attribute 'val'
调用父类的init方法有两种,第一种:
class Foo(object):
def __init__(self):
self.val = 1
class Foo2(Foo):
def __init__(self):
Foo.__init__(self)
print self.val
if __name__ == '__main__':
foo2 = Foo2()
第二种:
class Foo(object):
def __init__(self):
self.val = 1
class Foo2(Foo):
def __init__(self):
super(Foo2, self).__init__()
print self.val
if __name__ == '__main__':
foo2 = Foo2()
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
https://www.cndba.cn/dave/article/2151
版权声明:本文为博主原创文章,未经博主允许不得转载。