分类目录:《系统学习Python》总目录
除了将一个无self
方法进行特殊标记以外,我们有时还可以用一些不同的编码结构来达到类似的结果。例如,如果我们只想要调用一个不通过实例而访问类成员的函数,可能最简单的思路就是在类之外编写一个普通函数,而不是编写类方法。通过这种方式,调用中不会期待一个实例。例如,对文件的如下修改在Python3.X和Python2.X中都有效:
class Spam:
numInstances = 0
def __init__(self):
Spam.numInstances = Spam.numInstances + 1
def printNumInstances():
print(Spam.numInstances)
输入:
a = Spam()
b = Spam()
c = Spam()
printNumInstances()
输出:
3
因为类名称对简单函数而言是可读取的全局变量,这种方式可正常工作。此外,也要注意到函数名变成了全局变量,不过只针对这个模块而言;它不会和程序其他文件中的变量名冲突。
在Python中的静态方法之前,这一结构是通用的解决方案。由于Python已经把模块提供为命名空间划分工具,因此除非一个函数实现了对象的行为,否则我们总是可以认为不必把函数打包进类中。像这里这样的模块中的简单函数,实现了无实例类方法的大多数工作,并且已经与类关联起来,因为它们位于同一模块中。
遗憾的是,这种方法仍然不是理想的。其一,它给该文件的作用域添加了一个额外的名称,该名称唯一作用只是用来处理单个的类。其二,该数在代码结构上与类的联系很小;实际上,它的定义可能写在数百行代码之外的位置。可能更糟糕的是,由于它们位于类的命名空间之外的原因,像这样的简单函数不能通过继承定制,也就是说子类不能通过重新定义这样的一个函数来直接替代或扩展它。
我们也可以尝试像以往那样使用一个常规方法并总是通过一个实例调用它,从而使得这个例子以与Python版本无关的方式工作:
class Spam:
numInstances = 0
def __init__(self):
Spam.numInstances = Spam.numInstances + 1
def printNumInstances(self):
print(Spam.numInstances)
输入:
a = Spam()
b = Spam()
c = Spam()
a.printNumInstances()
Spam.printNumInstances(a)
Spam().printNumInstances()
输出:
3
3
4
遗憾的是,如果我们没有一个实例可用就无法使用该方法,然而如果产生一个实例来使用该方法又会改变类的数据,就像这里的最后一行。更好的解决方案可能是在类中把一个方法标记为不需要一个实例。
参考文献:
[1] Mark Lutz. Python学习手册[M]. 机械工业出版社, 2018.