延迟加载 lazy loading的粗略定义:第一次使用到某一个属性的时候再加载真正的内容到内存。
使用场景:
比如你需要在代码的某一步骤读一个巨大的CSV,耗时很长,所以你希望在用到此CSV内容的时候再进行具体读的操作,因为对于这个工程来说你可能有100个大型CSV待用,你需要先创建这些数据的对象,但具体的文件内容 attribute 你在没调用的时候不需要,你想先留一个空的 attribute。
一般人都喜欢在延迟加载中用@property
来实现 "用一个方法表示属性"这个目的。
实现:
在类的第一行创建一个为None的变量,在方法content
里调用真正的读写操作reading_csv
方法。这个 return 是关键,它的存在会使得最初的变量不再是None,因此 if 判断只会为真一次,如截图所示,第二次再调用 content 时就不会再运行reading_csv
方法了。
@property
可以把方法变为属性:content虽然为一个方法,但是当我调用的时候没有打结尾的括号,而是直接 g.content
,这样从人眼阅读角度要更为直白。不过你当然也可以就写一个普通的类方法,但是名字可能就要改为 get_content() 更准确一些。
使用@property
的好处还有一点,假如别人已经在工程其他地方用过你这个类的 value = object.content 了,你是后来才需要写这个动态加载的,那如果你再写一个 get_content() 方法,其他用过的地方全都要改成 value = get_content(), 哪怕你用属性的原名content来命名这个新方法,你也最起码要在其他使用的地方加上括号变成 value = object.content(),而使用@property来把一个原有的attribute改为方法不会影响其他地方的使用,因为语法不变。
import time
class my_csv():
_default_instance = None
def __init__(self, csv):
self.csv = csv
def reading_csv(self):
print("Taking 2s to read a big csv")
time.sleep(2)
return "I am the content"
@property
def content(self):
if not self._default_instance:
self._default_instance = self.reading_csv()
return self._default_instance