第三讲
一、如何在环状数据结构中管理内存
在python中,垃圾回收器通过引用计数来回收垃圾对象,当一个对象引用计数为0,或者只剩下弱引用时,这个对象会被释放。
1. 弱引用
弱引用不增加引用计数,使用弱引用访问对象得到对象引用。
In [1]: import weakref
In [2]: class B:
...: def __del__(self): # del方法在对象释放后触发
...: print("__del__")
...:
In [3]: b1 = B()
In [4]: b2 = weakref.ref(b1) # 弱引用语句
In [5]: b1 = None # 当b1不再指向创建的对象B(),引用计数为0,只剩下弱引用
__del__
我么来看一张图理解一下弱引用:
弱引用是可以踩实的,即变成正常引用:
In [1]: class A(object):
...: def __del__(self):
...: print('over')
...:
In [2]: import weakref
In [3]: a = A()
In [4]: b = weakref.ref(a)
In [5]: a
Out[5]: <__main__.A at 0x18b4856ea88>
In [6]: b
Out[6]: <weakref at 0x0000018B4850A638; to 'A' at 0x0000018B4856EA88>
In [7]: c = b() # 踩实过程,用一个新的变量来接收
In [8]: c
Out[8]: <__main__.A at 0x18b4856ea88>
In [9]: b
Out[9]: <weakref at 0x0000018B4850A638; to 'A' at 0x0000018B4856EA88>
弱引用踩实采用弱引用变量名加括号的方式。
2. 双链表
弱引用的应用场景之一为双链表,这里只做简单介绍:
变量head指向节点1,节点1右引用节点2,节点2右引用节点3。
节点3左弱引用节点2,节点2左弱引用节点1。
当变量head指向None时,节点1对象被释放,节点1的右引用节点2被释放,节点2的右引用节点3被释放。
目前只做了解。
二、通过实例方法名字的字符串调用方法
课程中这一节的内容主要还是回顾了几个重要方法:
getattr(x,'func_name',default=None) # 当x中不含有'func_name'命名的方法时,默然返回None。可通过第三个参数设置返回值
hasattr(x,'func_name')
map(func,Iterable)
不再罗列课程案例。
三、垃圾回收机制
1. 介绍
在Python程序运行的时候,会在内存中开辟一块空间,用于存放临时变量;当计算完成之后,就会将结果输出到永久性存储器中。如果数据量特别大,那内存空间管理不妥当的话就非常容易爆内存,程序可能直接终止。
在Python中,一切皆对象。所以,每一个变量,实际上都是对象的一个指针。所以,当这个对象的引用计数(指针数)为0的时候,说明它也变成了垃圾,需要被放到回收箱中。
2. os模块
os模块之前在基础课程已经有所涉及,主要是和操作系统交互的模块。
下面的案例中要接触一个新的模块内部方法:
os.getpid() # 获取当前进程id
3. psutil模块
psutil是一个跨平台库,能够轻松实现获取系统运行的进程和系统利用率(包括CPU、内存、磁盘、网络等)信息。它主要用来做系统监控,性能分析,进程管理。它实现了同等命令行工具提供的功能,如ps、top、lsofnetstat、ifconfig、who、df、kill、free、nice、ionice、iostat、iotop、uptime、pidof、tty、taskset、pmap等。目前支持32位和64位的Linux、Windows、OS X、FreeBSD和Sun Solaris等操作系统。这是一篇借鉴的文章,很详细。
我们来用一个案例来了解一下该模块的应用(内存信息调取),一个进程可以简单的理解为一个程序:
import os
import psutil
def show_info(start):
# 获取当前进程的id
pid = os.getpid()
# 获取当前进程对象
p = psutil.Process(pid)
# 返回该对象的内存消耗
info = p.memory_full_info()
# 获取进程独自占用的物理内存 换算单位为MB
memory = info.uss/1024/1024
print(f"{
start}一共占用{
memory:.2f<