Python 中的类方法和静态方法

简介: Python 中的类方法和静态方法

Python 中常见的方法分三种:实例方法、类方法和静态方法。实例方法最为常见,也最容易理解,而另外两种方法对于很多 Python 编程新手来说就不那么容易理解了,也比较容易用错。本文将对类方法和静态方法进行讲解,希望能够加深读者对这两种方法的理解。

实例方法

首先我们来简单回顾下 Python 中实例方法的定义:

1
2
3
classHello(object):
defsay_hello(self, name):
        print("hello", name)

如果要调用 Hello 类中定义的 say_hello 实例方法,需要先实例化一个实例对象 Hello(),再进行调用:

1
2
3
h = Hello()
h.say_hello("world")
# > hello world

类方法

回顾了实例方法,我们再来看下类方法如何使用。

类方法定义如下:

1
2
3
4
classHello(object):
    @classmethod
defprint_class_name(cls):
        print(cls.__name__)

对比实例方法,我们可以发现,类方法在定义上与实例方法有两处不同:首先在方法上面多了一个装饰器 @classmethod 标识这个方法为类方法,而不是实例方法;然后是方法参数由 self 变成了 cls 表示这个方法接收到的是一个类,而不是实例对象。

类方法的调用无需对类进行实例化,可以直接通过类来调用:

1
2
Hello.print_class_name()
# > Hello

通过 cls.__name__ 的方式我们打印了 Hello 类的类名。类方法中 cls 等价于 Hello,就像实例方法中 self 等价于实例对象 Hello() 一样。同 self 一样,cls 也是约定俗成的变量名,我们也可以不写成 cls 而写成任何我们喜欢的变量名。

类方法其实也可以通过实例对象进行调用:

1
2
3
h = Hello()
h.print_class_name()
# > Hello

得到的结果是一样的,其实当我们用实例对象调用类方法的时候,Python 解释器会自动帮我们获取到实例对象所属的类,然后将其当作参数传给类方法。

现在我们了解了类方法的语法,那么什么时候需要用到类方法呢?

类方法最常用的两个场景一个是类的继承,另一个是构造方法。

类的继承中使用类方法

首先来看如何在类继承中使用 cls 来做子类的名称绑定。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
classAnimal(object):
def__init__(self, name):
        self.name = name
    @classmethod
defshout(cls):
        print(f"{cls.__name__} shout")
classCat(Animal):
pass
classDog(Animal):
pass
cat = Cat("cat")
cat.shout()
# > Cat shout
dog = Dog("dog")
dog.shout()
# > Dog shout

这里我们定义一个动物类 Animal,其有一个 shout 方法用来打印哪种动物在叫。CatDog 类都继承自 Animal 类,当通过 Cat 类或 Dog 类的实例对象调用 shout 类方法时,就会分别打印出 Cat shoutDog shout。这样我们就实现了只需在父类中定义一个类方法,便可以在子类调用父类方法的时候根据类型的不同自动获取子类的类型名称。如果不使用类方法来定义,我们可能会写出如下代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
classAnimal(object):
def__init__(self, name):
        self.name = name
defshout(self):
        print("Animal shout")
classCat(Animal):
defshout(self):
        print("Cat shout")
classDog(Animal):
defshout(self):
        print("Dog shout")
cat = Cat("cat")
cat.shout()
# > Cat shout
dog = Dog("dog")
dog.shout()
# > Dog shout

虽然同样能实现功能,但这种实现大大增加了工作量,一是子类需要重写父类的方法,二是将类名 CatDog 都进行了硬编码,这种写法极不推荐。

我的开源项目 Todo List 教程models 部分就使用了这种思想来实现模型代码的复用,感兴趣的同学可以了深入解一下。

使用类方法实现构造方法

我们定义一个 Poerson 类,它的 __init__ 方法接收两个参数 firstlast 分别用来接收用户的 first_namelast_name,它还有一个 name 方法用来获取用户的全名。

1
2
3
4
5
6
7
8
9
10
11
classPerson(object):
def__init__(self, first, last):
        self.first_name = first
        self.last_name = last
defname(self):
return self.first_name + self.last_name
p = Person("江湖", "十年")
print(p.name())
# > 江湖十年

假设我们现在有一个爬虫程序,能够在网络上的某个网站中爬取到很多的用户名,这些用户名可能是 江湖-十年江湖_十年 或者 江湖十年 这三种形式。每一个用户名我们都需要实例化一个 Person 对象,由于用户名有三种形式,我们可以考虑使用一个函数来处理不同用户名,然后将处理后的用户名传给 Person 类来生成 Person 对象,可以写出如下代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
defgenerate_person(name):
if"-"in name:
        first, last = name.split("-")
elif"_"in name:
        first, last = name.split("_")
else:
        first, last = name, ""
return Person(first, last)
p1 = generate_person("江湖-十年")
p2 = generate_person("江湖_十年")
p3 = generate_person("江湖十年")
print(p1.name())
# > 江湖十年
print(p2.name())
# > 江湖十年
print(p3.name())
# > 江湖十年

这样,不管我们爬取到的用户名是哪种形式,都可以通过 generate_person 函数先预处理一下获取到 first_namelast_name,然后实例化 Person 对象进行返回。

其实有一种更好的方式对用户名进行预处理,那就是使用类方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
classPerson(object):
def__init__(self, first, last):
        self.first_name = first
        self.last_name = last
defname(self):
return self.first_name + self.last_name
    @classmethod
defnew(cls, name):
if"-"in name:
            first, last = name.split("-")
elif"_"in name:
            first, last = name.split("_")
else:
            first, last = name, ""
return cls(first, last)
p1 = Person.new("江湖-十年")
p2 = Person.new("江湖_十年")
p3 = Person.new("江湖十年")
print(p1.name())
# > 江湖十年
print(p2.name())
# > 江湖十年
print(p3.name())
# > 江湖十年

我们将 generate_person 函数移动到 Person 类内部作为一个类方法,并将其重命名为 new,最后通过 return 一个 cls 实例来返回 Person 实例对象。

熟悉 Java 等静态语言的同学应该很容易想到,这个类方法 new 实际上就是其他编程语言中的 构造函数。所以我们也就通过类方法的形式,在 Python 中实现了构造方法(函数)。

相比于使用 generate_person 函数生成 Person 实例对象,使用 new 构造方法的好处是代码封装性更强,也更容易理解。构造函数本就应该属于类的一部分,而不是单独写一个函数来实现。

爬虫框架 ScrapyItem Pipeline 中有一个叫 from_crawler 的类方法,实际上就是 Scrapy 提供给我们的一个构造方法。

静态方法

接下来我们一起看下静态方法在 Python 中的应用。

静态方法定义如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
from datetime import datetime
classHello(object):
def__init__(self, name):
        self.name = name
defsay_hello(self, create_time):
        print(f"hello {self.name} - {self.process_datetime(create_time)}.")
    @staticmethod
defprocess_datetime(dt):
return dt.strftime("%Y-%m-%d %H:%S:%M")
h = Hello("tim")
h.say_hello(datetime.now())
# > hello tim - 2099-01-07 13:14:15.

Hello 类的 process_datetime 方法即为一个静态方法,可以看到静态方法的定义跟一个普通函数区别不大,只是在上面多了一个 @staticmethod 装饰器。这个函数作用是对日期时间类型做格式化操作,返回我们想要的 str 类型的日期时间,在调用 say_hello 时可以通过参数传递进来一个 create_time,我们想把它打印到 hello name 问候语句的后面,于是调用了 process_datetime 静态方法来对日期时间进行格式化处理。

这个示例代码比较简单,所以优势不是非常明显,但假设我们的 process_datetime 方法有大量逻辑要处理,这样单独提取出来一个静态方法,会比把所有逻辑写在 say_hello 方法内部更加合理,因为这样的话代码可读性会更高,并且可复用。

对比来看,相较于类方法来说,静态方法更加简单,使用场景也更少。

Django 框架源码中有多处用到了 staticmethod,比如在 modelsRegisterLookupMixin 类下有一个 merge_dicts 方法作用就使用了静态方法将多个 dict 进行合并操作,这个操作不涉及类属性或实例属性的引用,所以使用静态方法比较合适。

以上,我介绍了 Python 中类方法和静态方法的定义和使用,希望你在写代码的时候能够更加得心应手,我只对常见用法做了总结,更多用法还是需要你在实践中更加深入的体会。同时我也在每个示例讲解的下面贴上了开源项目中对不同方法的使用,通过阅读优秀开源项目的源码,也能够加深我们对“最佳实践”的理解。

目录
打赏
0
0
0
0
4
分享
相关文章
Python数值方法在工程和科学问题解决中的应用
本文探讨了Python数值方法在工程和科学领域的广泛应用。首先介绍了数值计算的基本概念及Python的优势,如易学易用、丰富的库支持和跨平台性。接着分析了Python在有限元分析、信号处理、优化问题求解和控制系统设计等工程问题中的应用,以及在数据分析、机器学习、模拟建模和深度学习等科学问题中的实践。通过具体案例,展示了Python解决实际问题的能力,最后总结展望了Python在未来工程和科学研究中的发展潜力。
|
2月前
|
Python技术解析:了解数字类型及数据类型转换的方法。
在Python的世界里,数字并不只是简单的数学符号,他们更多的是一种生动有趣的语言,用来表达我们的思维和创意。希望你从这个小小的讲解中学到了有趣的内容,用Python的魔法揭示数字的奥秘。
83 26
|
2月前
|
在VScode环境下配置Python环境的方法
经过上述步骤,你的VSCode环境就已经配置好了。请尽情享受这扇你为自己开启的知识之窗。如同你在冒险世界中前行,你的探索之路只有越走越广,你获得的知识只会越来越丰富,你的能力只会越来越强。
217 37
解决Python requests库POST请求参数顺序问题的方法。
总之,想要在Python的requests库里保持POST参数顺序,你要像捋顺头发一样捋顺它们,在向服务器炫耀你那有条不紊的数据前。抓紧手中的 `OrderedDict`与 `json`这两把钥匙,就能向服务端展示你的请求参数就像经过高端配置的快递包裹,里面的商品摆放井井有条,任何时候开箱都是一种享受。
57 10
|
2月前
|
Python 中__new__方法详解及使用
__new__ 是 Python 中用于创建类实例的静态方法,在实例化对象时优先于 __init__ 执行。它定义在基础类 object 中,需传递 cls 参数(表示当前类)。__new__ 可决定是否使用 __init__ 方法或返回其他对象作为实例。特性包括:1) 在实例化前调用;2) 始终为静态方法。示例中展示了其用法及 Python2 和 Python3 的差异,强调了参数处理的不同。
112 10
python__init__方法笔记
本文总结了Python中`__init__`方法的使用要点,包括子类对父类构造方法的调用规则。当子类未重写`__init__`时,实例化会自动调用父类的构造方法;若重写,则需通过`super()`或直接调用父类名称来显式继承父类初始化逻辑。文中通过具体代码示例展示了不同场景下的行为及输出结果,帮助理解类属性与成员变量的关系,以及如何正确使用`super()`实现构造方法的继承。
103 9
[oeasy]python093_find方法_指数为负数_index_实际效果
本文介绍了Python中`find`方法与索引(index)的使用,包括负数索引的实际效果。回顾了`eval`函数的应用,并强调类名如`str`、`int`、`list`不可用作变量名以避免覆盖。通过示例解析了负数索引在字符串和列表中的作用,以及`index`方法的三个参数(value、start、stop)的用法。同时对比了`index`和`find`方法的区别:`index`找不到子串时抛出`ValueError`,而`find`返回-1。最后总结了正负索引的使用场景及两者的特性,提供了相关学习资源链接。
311 8
|
3月前
|
解决Python报错:DataFrame对象没有concat属性的多种方法(解决方案汇总)
总的来说,解决“DataFrame对象没有concat属性”的错误的关键是理解concat函数应该如何正确使用,以及Pandas库提供了哪些其他的数据连接方法。希望这些方法能帮助你解决问题。记住,编程就像是解谜游戏,每一个错误都是一个谜题,解决它们需要耐心和细心。
181 15
uv安装python及其依赖的加速方法
国内在使用uv的时候,可能会涉及到装python的速度太慢的问题,为了解决这个问题,可以使用`UV_PYTHON_INSTALL_MIRROR`这个环境变量。除此以外,对于多人协作场景,`UV_CACHE_DIR`也是一个有用的环境变量。本文会介绍这两个变量。
2213 10
|
2月前
|
Python 中__new__方法详解及使用
`__new__` 是 Python 中的一个特殊方法,用于控制对象的创建过程,在 `__init__` 之前执行。它是类的静态方法,负责返回一个实例。如果 `__new__` 不返回对象,`__init__` 将不会被调用。本文详细介绍了 `__new__` 的作用、特性及与 `__init__` 的区别,并通过实例演示了其在单例模式中的应用,同时对比了 Python2 和 Python3 中的写法差异。

推荐镜像

更多
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问