Python学习笔记(4)——类和方法

类和对象,作为面向对象编程的核心,在所有OOP语言里面都是重头戏,Python也不例外。
Python中的类和对象定义如下:

class foo:
    pass


f = foo()

class关键字开头,定义了一个foo类,并使用foo()实例化成对象f,这里没有定义任何类内变量或者方法,所以使用pass关键字。

Python类的继承

出人意料的是,Python是支持多继承的,作为一门相对年轻的语言,这个特性可不多见了,毕竟从java开始,多继承就被认为是一种不好的设计,如果你读过《Effective C++》,那么你应该知道有些时候连继承都不是一种好的设计。
Anyway,我没有资格对Python的设计妄加评判,总之,Python的继承语法如下:

class Human:
    pass


class Father(Human):
    pass


class Mother(Human):
    pass


class Child(Father, Mother):
    pass

另外,当一个类没有指定任何基类时,它默认继承自内置类型object
由于Python不存在protected约束,所以在实现多态时,有时需要你去进行一些约定,而无法从语法上保证安全性。
_前缀表示该成员是module级别的私有成员,无法被别的module访问,但是module内部是可以随意访问的。
__前缀则可以定义一个private的成员(变量跟方法都可以),派生类无法访问该成员,但是可以通过“名字改编”(name mangling)的技术间接访问:

class Human:
    __weight = 100

    def __say(self, arg):
        print(arg)


class Father(Human):
    pass


class Mother(Human):
    pass


class Child(Father, Mother):
    pass


print(Child().__weight) # 非法
Child().__say('hello') # 非法
print(Child()._Human__weight) # 合法,name mangling
Child()._Human__say('hello') # 合法,name mangling

这里使用的别名是指向源地址的,也就是说你对它做的赋值等行为都是有效的。
另外,当一个成员同时使用__作为前缀和后缀时,它不表示这个成员是个私有成员,而表示这是一个“魔法成员”,通常这是内建属性,当然你也可以自定义如果你不怕把自定义成员跟内建成员弄混的话(或者你本来就想扩展一下)。

Python类的内置属性

Python中的类,使用类名时本身也是一个type类的对象:

class foo:
    pass


print(type(foo))

可以看到foo的类型为type
那么类本身作为对象,它就具有内置的属性,类内置属性很多很多,其中几个特殊属性(官网说的special attributes)如下:

属性 作用
__name__ 类名
__module__ 类定义所在的模块名
__dict__ 类命名空间的字典
__bases__ 类的所有基类
__doc__ 类文档
__annotations__ 类注解

可以看到,有些属性跟Python学习笔记(3)——函数中提到的callable类型的内置属性相同,作用也是一样的。
其中,__dict__包含了类的所有变量和方法,callable类型也有__dict__,但它通常是空的。
__bases__则包含了该类继承的所有基类(不包含基类的基类),如上一节的Human,它的__bases__(<class 'object'>,),而Child__bases__则是(<class '__main__.father'>, <class '__main__.mother'>)

Python类方法的定义和调用

Python类内方法的定义如下:

class foo:
    def bar(self):
        pass

跟普通函数的定义差不多,但是类内方法第一个参数必须是self(当然名字可以自定,self是某种默认的规则),该参数的作用类似java的this,不过Python需要显式定义。
当然你可以不定义这个参数:

class foo:
    def bar():
        pass

此时,bar便不再是一个普通的方法,你无法使用实例化的对象去调用它,因为实例化对象的调用形式foo().bar()会把对象本身作为第一个参数传递进去,如此调用你会获得一个错误:TypeError: bar() takes 0 positional arguments but 1 was given
但是,使用类名去调用该函数是可以的:foo.bar(),这个形式相信大家都很熟悉了,这不就是静态方法吗?

Python静态方法、类方法

Python的静态方法定义如下:

class foo:
    @staticmethod
    def bar():
        pass

可以看到这里我们又用上了装饰器@staticmethod,看过前一章的朋友们对这个语法应该很熟悉了,在builtins.pyi文件中定义了这个装饰器类staticmethod(有的同学可能有疑问,装饰器不是函数吗?怎么类也可以当装饰器?这个后面再解释),具体作用就是给被装饰的方法添加了一些属性,所以事实上调用起来,加不加@staticmethod都可以成功执行,当然为了代码可读性一般都是加的。
除了静态方法之外,Python还有一种特殊的方法也是用类名调用的,被称作类方法(classmethod)

class foo:
    @classmethod
    def bar(cls):
        pass

虽然调用形式都是foo.bar(),但是classmethod多一个cls参数,类似普通方法的selfcls会将类本身作为一个参数传进去,如果你需要对类本身的属性进行操作,那么你应该使用classmethod而非staticmethod,从这个角度看,classmethod更加接近C++的类静态函数。

Python虚函数

Python没有虚函数和纯虚函数,所以你也无法实现接口,当然你可以用某种手段实现类似的功能,比如装饰器:

def virtual(func):
    def wrapper(self):
        raise IOError('Not implement')
    return wrapper


class interface:
    @virtual
    def virtual_func(self): pass


class implement(interface):
    def virtual_func(self):
        print('hello')

以上写法,调用interface().virtual_func()会抛出一个IOError,而调用implement().virutal_func()则可以打印hello。

Python类的内置方法

Python类的主要内置方法如下:

方法 作用
__new__ 创建新实例时调用
__init__ __new__调用以后,返回结果之前调用
__del__ 在实例销毁时调用
__repr__ 完整的对象名
__str__ 更加可读的对象名,同str(object)
__bytes__ __str__的字节形式,同bytes(object),但是我在3.8.0下无法使用,官网的3.8是有这个方法的
__format__ '{}'.format(object)时调用,你需要在内部将该对象作为字符串时应该输出什么给定义好
__lt__ __le__ __eq__ __ne__ __gt__ __ge__ 分别对应< <= == != > >=
__hash__ hash(object),使用hash()函数计算对象哈希值时的输出
__bool__ 该对象进行逻辑运算时的值
__getattribute__ 获取对象内部属性,以属性名为参数,可以实现某种反射机制吧
__getattr__ 当其它获取属性的方法抛出AttributeError时调用
__setattr__ 当给属性赋值时调用
__delattr__ 当删除属性时调用,即调用del关键字删除对象内部变量时
__dir__ dir(object)

别的不多说了,__getattr__比较有意思,通常我们获取一个对象的成员变量,若写错变量名,会抛出一个AttributeError,但是如果你重写了__getattr__,则可以自定义这一行为:

class foo:
    def __getattr__(self,name):
        return 'variable did not define'


print(foo().bar)

如上代码会打印字符串'variable did not define'而并不会报错,当然我们一般不希望变量在未定义之前就使用,所以最好还是抛出错误中止程序比较好。
__del__就是通常所说的析构函数,由于Python是自动GC的,和java相似,一个健康的设计不应该把释放资源的操作放进__del__,而是实现__enter__ __exit__方法并使用with关键字更加安全。
__new____init__则构成了构造函数__new__是一个特殊的classmethod,它没有用@classmethod装饰,但是也接收类作为第一个参数,每当你实例化一个对象时,都会调用__new__,默认返回cls的一个实例,当然你可以通过修改__new__来使得它返回一个别的东西,当然一般不推荐这么做:

class foo:
    def __new__(cls):
        return 'hello'


print(foo())

类似上面的代码,将使得foo()对象变成一个'hello'字符串。
也许可以应用于过于冗长的函数或者类的别名?

class waibibabububububububu:
    def say_hi(self):
        print('hi')


class foo:
    def __new__(cls):
        return waibibabububububububu.__new__(waibibabububububububu)


foo().say_hi()

但是装饰器可以起到一样的效果,所以总的来说,没有必要去对__new__的返回类型作出修改,但是修改它的行为有可能是需要的,比如你需要实现一个单例模式时,这个以后再说。
另外可以看到,由于__new__没有使用@classmethod装饰,它并不会默认接收它的调用者作为第一个参数,所以这里显式地传递了一个类进去。
__init__则是一个普通的成员方法,它默认接收调用它的实例self作为第一个参数,__init__的调用时机是在__new__内部创建实例时,也就是__new__返回之前就调用了__init__
对于C++程序员来说__init__更接近于构造函数,大多数时候我们也只需要重写__init__,需要注意的是与C++不同,父类的__init__必须被显式调用才能执行:

class foo:
    def __init__(self):
        print('foo')


class bar(foo):
    def __init__(self):
        super().__init__()
        print('bar')

Python装饰器类

在函数一章中我们了解了用函数写的装饰器,而前面的staticmethod classmethod则是用类实现的装饰器,这是怎么做到的呢?
我们已经知道Python的函数本身也是一个callable对象,那么当一个普通对象具有和callable对象类似的属性时,这个对象也就可以作为函数使用了:

class decorator:

    def __init__(self, func):
        self.__func = func

    def __call__(self, *args, **kwds):
        print('decorate') 
        if self.__func is not None:
            self.__func(*args, **kwds)


@decorator
def foo(arg):
    print(arg)


foo('hello')

这里用到了Python类的一个内置方法:__call__,当你实现了一个类的__call__方法,那么这个类的对象就会成为一个callabe对象,可以使用后缀()的方式调用它的__call__方法。
而上面的装饰器严格等价于foo = decorator(foo),即将foo类作为参数初始化了一个decorator对象,并将foo重新赋值为该对象,打印type(foo),可以发现它的类型是<class '__main__.decorator'>
带参的装饰器和复数个装饰器的例子可以参考函数一章,这里就不作展开了。只要记住@装饰器跟展开式是完全等价的,并且明白对象跟函数本质上是一样的,那么一切就都很好理解了。
下面是烧脑时间:

class decorator:

    def __init__(self, func):
        self.__func = func

    def __call__(self, *args, **kwds):
        def wrapper(*_arg):
            print(*_arg)
            if self.__func is not None:
                self.__func(*args, **kwds)
        return wrapper


def f1(arg):
    def wrapper(class_decorator):
        return class_decorator(arg)
    return wrapper


@f1('world')
@decorator
def foo(arg):
    print(arg)


foo('hello ')

以上代码会打印hello world,请整理出上面两个装饰器的调用过程。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容