@classmethod 和 @staticmethod 方法

本篇文章,我将介绍类方法 classmethod 和静态方法 staticmethod 以及它们之间的不同。类方法和静态方法都是用装饰器来定义一个方法使它成为一个类方法或静态方法。如果对装饰器不熟的请参阅我写的有关《python装饰器》这篇文章。

成员方法(实例方法)、静态方法和类方法

实例方法

类中最常用的方法是实例方法, 即通过通过实例作为第一个参数的方法。
举个例子,一个基本的实例方法就向下面这个:

class Human(object):
    human_count = 0
    
    def __init__(self, name):
        self.name = name
        Human.human_count += 1
    
    def print_name(self):
        print("My name is: ", self.name)

h = Human("Adam")
h.print_name()

运行结果为:

My name is: Adam

这里的 print_name 方法就是个实例方法。该类方法的第一个参数为调用其的实例对象。传入实例对象作为参数是 python 帮我们自动完成的。

类方法和静态方法的引入

在这里向说明两个问题。首先是 python 的机制问题。其次是python 没法像 c++ 那样进行方法的重载(接受的参数个数或类型不同就可以重载一个方法,即用同一个方法名)的问题。为了更好的说明这两个问题,我们下面做两个实验:

实验1:

class Human(object):
    human_count = 0
    
    def __init__(self, name):
        self.name = name
        Human.human_count += 1
    
    def print_name(self):
        print("My name is: ", self.name)
    
    def print_hello():
        print("Hello, nice world!")
    

h = Human("Adam")
h.print_name()
h.print_hello()

在实验1中我们写了一个print_hello方法,其不接受任何参数。当我们在实例化一个 Human 对象 h,并且想通过 h 来调用 print_hello 这个方法。但是我们发现,上述代码的输出为:

TypeError                                 Traceback (most recent call last)
<ipython-input-98-8b96112221b9> in <module>()
     15 h = Human("Adam")
     16 h.print_name()
---> 17 h.print_hello()

TypeError: print_hello() takes no arguments (1 given)

我们发现提示信息告诉我们,print_hello()函数是不需要参数的(这和咱们写这个函数的初衷是一样的),但是该方法却被传入了一个参数,因此报错。从这里我们就知道了 python 的机制。凡是通过类实例调用的方法默认给方法传入了一个参数,也就是该实例本身。也就是之前咱们写实例方法时,每个方法的第一个参数 self

实验2:

class Human(object):
    human_count = 0
    
    def __init__(self, name):
        self.name = name
        Human.human_count += 1
    
    def print_name(self):
        print("My name is: ", self.name)
    
    def print_name(self, a):
        print("This is the other print_name method with parameter a: ", a)
    

h = Human("Adam")
h.print_name()
h.print_name(11)

按照我们的思路,我们是想按照 c++ 重载函数的思想来写一个同名的 print_name 方法,除了实例本身外还接受一个参数 a。然后分别用实例 h 调用两个函数。
输出结果:

TypeError                                 Traceback (most recent call last)
<ipython-input-99-ef128668b1a3> in <module>()
     14 
     15 h = Human("Adam")
---> 16 h.print_name()
     17 h.print_name(11)

TypeError: print_name() takes exactly 2 arguments (1 given)

我们发现,输出结果和我们的预想不一样。发现后面写的方法 print_name(self, a) 已经把之前的同名方法覆盖。也就是说前面的那个方法不存在了。此时,类中的实例方法 print_name 只剩下接受两个参数的那个。也就是只剩下 print_name(self, a) 这个方法。因此报错信息说 print_name() 方法需要两个参数,而我们只给了一个(即实例本身)。
通过上述两个实验,我们说明白了 python 调用类内函数的机制以及无法重载函数的问题。这里面特别是因为我们的 python 机制存在,使得我们需要想办法使得一些不需要实例本身作为参数的方法得以创建和正常使用。

类方法

当我们需要和类直接进行交互,而不需要和实例进行交互时。类方法就是最好的选择。但是之前所介绍的 python 机制问题,我们知道我们通过实例调用方法时都会默认传入实例本身作为第一个参数。而如今,类方法不需要和实例进行交互,自然也就不需要传入实例本身。再则,类方法需要和类进行交互,所以如果有种方法能够在每次调用时都传入类对象就好了。而这个方法就是使用 @classmethod 来装饰我们的方法。如下代码:

class Human(object):
    human_count = 0
    
    def __init__(self, name):
        self.name = name
        Human.human_count += 1
    
    def print_name(self):
        print("My name is: ", self.name)
    
    @classmethod
    def number(cls):
        print("The number of people: ", cls.human_count)
    

h = Human("Adam")
h.print_name()
h.number()
Human.number()

代码输出为:

My name is:  Adam
The number of people:  1
The number of people:  1

经过 @classmethod 装饰后,我们可以直接通过实例来调用类方法,也可以通过类名来调用。而类方法的第一个参数为类对象本身(按照习惯或约定一般写为 cls,代表 class_object 本身)。这样我们就能在类方法中,轻松和类进行交互:访问类本身的属性以及类本身的方法。
我们还可以利用这个特性轻松通过类方法来创建实例,而这个功能就类似于 c++ 中的重载(从外部看起来像是重载了构造函数):

class Human(object):
    human_count = 0
    
    def __init__(self, name):
        self.name = name
        Human.human_count += 1
    
    def print_name(self):
        print("My name is: ", self.name)
    
    @classmethod
    def number(cls):
        print("The number of people: ", cls.human_count)
    
    @classmethod
    def through_list(cls, lst):
        new_member = list()
        for i in range(len(lst)):
            new_member.append(cls(lst[i]))
        return new_member
    

h = Human("Adam")
h.print_name()
h.number()
Human.number()
lst = ["Lucy", "Stack", "Lala"]
human_lst = h.through_list(lst)
for i in range(len(human_lst)):
    print(human_lst[i].name)

输出结果:

My name is:  Adam
The number of people:  1
The number of people:  1
Lucy
Stack
Lala

静态方法

我们经常会遇到一些这样的情况:我们需要一些方法。这些方法和类相关,但是又不需要类和实例中的任何信息、属性等等。如果把这些方法写到类外面,这样就把和类相关的代码分散到类外,使得之后对于代码的理解和维护都是巨大的障碍。而静态方法就是用来解决这一类问题的。举个例子,比如我们现在要设置一个环境变量 RESET。还要写一个方法,这个方法将检测 RESET 变量的值。当变量值r为真时我们需要打印一句话:“The reset is on!”。这个方法和类相关,但又不需要访问任何和类或者实例相关的属性。所以我们希望把它绑定在类内,而不是在类外写一个这样的方法,然后调用。怎么办呢?直接上代码:

RESET = True

class Human(object):
    human_count = 0
    
    def __init__(self, name):
        self.name = name
        Human.human_count += 1
    
    def print_name(self):
        print("My name is: ", self.name)

    @staticmethod
    def whether_reset():
        if RESET:
            print("The reset is on!")

h = Human("Adam")
h.whether_reset()

运行结果:

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

推荐阅读更多精彩内容

  • Python 面向对象Python从设计之初就已经是一门面向对象的语言,正因为如此,在Python中创建一个类和对...
    顺毛阅读 4,167评论 4 16
  • 一、快捷键 ctr+b 执行ctr+/ 单行注释ctr+c ...
    o_8319阅读 5,728评论 2 16
  • 一、Python简介和环境搭建以及pip的安装 4课时实验课主要内容 【Python简介】: Python 是一个...
    _小老虎_阅读 5,618评论 0 10
  • 许是天气过于炎热,阳光越来越刺眼,每天早晨起来的时间越来越早,晃眼的光线,透过窗户照进来,朦朦胧胧,周一上班就要开...
    滴答滴答的小园丁阅读 202评论 0 0
  • 听过一句话:不是蝴蝶飞不过沧海,只是沧海那头没有等待! 很多事情,我们完不成、做不到,只是因为沧海的那头没有等待!...
    岑陌阅读 167评论 0 0