签到成功

知道了

CNDBA社区CNDBA社区

python中 类(Class)的 实例方法、静态方法、类方法、类变量和实例变量 说明

2017-08-20 10:19 3197 0 原创 Python
作者: dave

相关概念区别说明:
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
<type 'classobj'>
<type 'instance'>
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

4 super
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()


http://www.cndba.cn/dave/article/2151 http://www.cndba.cn/dave/article/2151 http://www.cndba.cn/dave/article/2151 http://www.cndba.cn/dave/article/2151 http://www.cndba.cn/dave/article/2151 http://www.cndba.cn/dave/article/2151
http://www.cndba.cn/dave/article/2151
http://www.cndba.cn/dave/article/2151
http://www.cndba.cn/dave/article/2151
http://www.cndba.cn/dave/article/2151

版权声明:本文为博主原创文章,未经博主允许不得转载。

用户评论
* 以下用户言论只代表其个人观点,不代表CNDBA社区的观点或立场
dave

dave

关注

人的一生应该是这样度过的:当他回首往事的时候,他不会因为虚度年华而悔恨,也不会因为碌碌无为而羞耻;这样,在临死的时候,他就能够说:“我的整个生命和全部精力,都已经献给世界上最壮丽的事业....."

  • 2261
    原创
  • 3
    翻译
  • 578
    转载
  • 191
    评论
  • 访问:8030956次
  • 积分:4346
  • 等级:核心会员
  • 排名:第1名
精华文章
    最新问题
    查看更多+
    热门文章
      热门用户
      推荐用户
        Copyright © 2016 All Rights Reserved. Powered by CNDBA · 皖ICP备2022006297号-1·

        QQ交流群

        注册联系QQ