一般来说在Python 中,为了解决内存泄漏问题,采用了对象引用计数,并基于引用计数实现自动垃圾回收。由于Python 有了自动垃圾回收功能,就造成了不少初学者误认为自己从此过上了好日子,不必再受内存泄漏的骚扰了。但如果仔细查看一下Python文档对 __del__() 函数的描述,就知道这种好日子里也是有阴云的。下面摘抄一点文档内容如下:

Some common situations that may prevent the reference count of an object from going to zero include: circular references between objects (e.g., a doubly-linked list or a tree data structure with parent and child pointers); a reference to the object on the stack frame of a function that caught an exception (the traceback stored in sys.exc_traceback keeps the stack frame alive); or a reference to the object on the stack frame that raised an unhandled exception in interactive mode (the traceback stored in sys.last_traceback keeps the stack frame alive).

可见,有 __del__() 函数的对象间的循环引用是导致内存泄漏的主凶。

另外需要说明:对没有 __del__() 函数的 Python 对象间的循环引用,是可以被自动垃圾回收掉的。

如何知道一个对象是否内存泄漏了呢?

方法一、当你认为一个对象应该被销毁时(即引用计数为 0),可以通过 sys.getrefcount(obj) 来获取对象的引用计数,并根据返回值是否为 0 来判断是否内存泄漏。如果返回的引用计数不为 0,说明在此刻对象 obj 是不能被垃圾回收器回收掉的。

方法二、也可以通过 Python 扩展模块 gc 来查看不能回收的对象的详细信息。

首先,来看一段正常的测试代码:

#--------------- code begin --------------

# -*- coding: utf-8 -*-

import gc

import sys

class CGcLeak(object):

def __init__(self):

self._text = '#'*10

def __del__(self):

pass

def make_circle_ref():

_gcleak = CGcLeak()

# _gcleak._self = _gcleak # test_code_1

print '_gcleak ref count0:%d' % sys.getrefcount(_gcleak)

del _gcleak

try:

print '_gcleak ref count1:%d' % sys.getrefcount(_gcleak)

except UnboundLocalError:

print '_gcleak is invalid!'

def test_gcleak():

# Enable automatic garbage collection.

gc.enable()

# Set the garbage collection debugging flags.

gc.set_debug(gc.DEBUG_COLLECTABLE | gc.DEBUG_UNCOLLECTABLE | /

gc.DEBUG_INSTANCES | gc.DEBUG_OBJECTS)

print 'begin leak test...'

make_circle_ref()

print 'begin collect...'

_unreachable = gc.collect()

print 'unreachable object num:%d' % _unreachable

print 'garbage object num:%d' % len(gc.garbage)

if __name__ == '__main__':

test_gcleak()

在 test_gcleak() 中,设置垃圾回收器调试标志后,再用 collect() 进行垃圾回收,最后打印出该次垃圾回收发现的不可达的垃圾对象数和整个解释器中的垃圾对象数。

gc.garbage 是一个 list 对象,列表项是垃圾收集器发现的不可达(即是垃圾对象)、但又不能释放(即不能回收)的对象。文档描述为:A list of objects which the collector found to be unreachable but could not be freed (uncollectable objects).

通常,gc.garbage 中的对象是引用环中的对象。因为 Python 不知道按照什么样的安全次序来调用环中对象的 __del__() 函数,导致对象始终存活在 gc.garbage 中,造成内存泄漏。如果知道一个安全的次序,那么就打破引用环,再执行 del gc.garbage[:] ,以清空垃圾对象列表。

上段代码输出为(#后字符串为笔者所加注释):

#-----------------------------------------

begin leak test...

# 变量 _gcleak 的引用计数为 2.

_gcleak ref count0:2

# _gcleak 变为不可达(unreachable)的非法变量.

_gcleak is invalid!

# 开始垃圾回收

begin collect...

# 本次垃圾回收发现的不可达的垃圾对象数为 0.

unreachable object num:0

# 整个解释器中的垃圾对象数为 0.

garbage object num:0

#-----------------------------------------

由此可见 _gcleak 对象的引用计数是正确的,也没有任何对象发生内存泄漏。

如果不注释掉 make_circle_ref() 中的 test_code_1 语句:

_gcleak._self = _gcleak

也就是让 _gcleak 形成一个自己对自己的循环引用。再运行上述代码,输出结果就变成:

#-----------------------------------------

begin leak test...

_gcleak ref count0:3

_gcleak is invalid!

begin collect...

# 发现可以回收的垃圾对象: 地址为 012AA090,类型为 CGcLeak.

gc: uncollectable

gc: uncollectable

unreachable object num:2

#!! 不能回收的垃圾对象数为 1,导致内存泄漏!

garbage object num:1

#-----------------------------------------

可见 对象发生了内存泄漏!!而多出的 dict 垃圾就是泄漏的 _gcleak 对象的字典,打印出字典信息为:

{'_self': <__main__.cgcleak object at>, '_text': '##########'}

除了对自己的循环引用,多个对象间的循环引用也会导致内存泄漏。简单举例如下:

#--------------- code begin --------------

class CGcLeakA(object):

def __init__(self):

self._text = '#'*10

def __del__(self):

pass

class CGcLeakB(object):

def __init__(self):

self._text = '*'*10

def __del__(self):

pass

def make_circle_ref():

_a = CGcLeakA()

_b = CGcLeakB()

_a._b = _b # test_code_2

_b._a = _a # test_code_3

print 'ref count0:a=%d b=%d' % /

(sys.getrefcount(_a), sys.getrefcount(_b))

# _b._a = None # test_code_4

del _a

del _b

try:

print 'ref count1:a=%d' % sys.getrefcount(_a)

except UnboundLocalError:

print '_a is invalid!'

try:

print 'ref count2:b=%d' % sys.getrefcount(_b)

except UnboundLocalError:

print '_b is invalid!'

#--------------- code end ----------------

这次测试后输出结果为:

#-----------------------------------------

begin leak test...

ref count0:a=3 b=3

_a is invalid!

_b is invalid!

begin collect...

gc: uncollectable

gc: uncollectable

gc: uncollectable

gc: uncollectable

unreachable object num:4

garbage object num:2

#-----------------------------------------

可见 _a,_b 对象都发生了内存泄漏。因为二者是循环引用,垃圾回收器不知道该如何回收,也就是不知道该首先调用那个对象的 __del__() 函数。

采用以下任一方法,打破环状引用,就可以避免内存泄漏:

1.注释掉 make_circle_ref() 中的 test_code_2 语句;

2.注释掉 make_circle_ref() 中的 test_code_3 语句;

3.取消对 make_circle_ref() 中的 test_code_4 语句的注释。

相应输出结果变为:

#-----------------------------------------

begin leak test...

ref count0:a=2 b=3 # 注:此处输出结果视情况变化.

_a is invalid!

_b is invalid!

begin collect...

unreachable object num:0

garbage object num:0

#-----------------------------------------

结论:Python 的 gc 有比较强的功能,比如设置 gc.set_debug(gc.DEBUG_LEAK) 就可以进行循环引用导致的内存泄露的检查。如果在开发时进行内存泄露检查;在发布时能够确保不会内存泄露,那么就可以延长 Python 的垃圾回收时间间隔、甚至主动关闭垃圾回收机制,从而提高运行效率。

python的gc模块_Python的内存泄漏及gc模块的使用分析相关推荐

  1. Android—内存泄漏、GC及LeakCanary源码解析

    内存抖动:内存频繁的分配和回收,频繁的GC会导致UI卡顿,严重的时候导致OOM. 内存泄露:程序在向系统申请分配内存空间后(new),在使用完毕后未释放.结果导致一直占据该内存单元,我们和程序都无法再 ...

  2. python 内存溢出_Python之内存泄漏和内存溢出

    一.内存泄漏 像Java程序一样,虽然Python本身也有垃圾回收的功能,但是同样也会产生内存泄漏的问题. 对于一个用 python 实现的,长期运行的后台服务进程来说,如果内存持续增长,那么很可能是 ...

  3. python怎么清理垃圾_python清理内存

    广告关闭 腾讯云11.11云上盛惠 ,精选热门产品助力上云,云服务器首年88元起,买的越多返的越多,最高返5000元! 语言的内存管理是语言设计的一个重要方面. 它是决定语言性能的重要因素. 无论是c ...

  4. 【Android 内存优化】垃圾回收算法 ( 内存优化总结 | 常见的内存泄漏场景 | GC 算法 | 标记清除算法 | 复制算法 | 标记压缩算法 )

    文章目录 一. 内存优化总结 二. 常见的内存泄漏场景 三. 内存回收算法 四. 标记-清除算法 ( mark-sweep ) 五. 复制算法 六. 标记-压缩算法 一. 内存优化总结 内存泄漏原理 ...

  5. python 异常处理模块_Python Exceptions 异常处理全过程以及异常模块详细解析

    1.异常的定义 程序运行时检测到错误而导致解释器无法继续执行,反而出现一些错误的提示 2.捕捉异常全过程(重点) try: print("可能产生异常的代码") except Ex ...

  6. python modbus类封装_Python 中引入一个文件,模块的概念

    Python 提供了强大的模块支持,主要体现在,不仅 Python 标准库中包含了大量的模块(称为标准模块),还有大量的第三方模块,开发者自己也可以开发自定义模块. 通过这些强大的模块可以极大地提高开 ...

  7. python pywinauto 单击鼠标_Python 基础(十):模块与包

    1 简介 1.1 模块 Python 中一个以 .py 结尾的文件就是一个模块,模块中定义了变量.函数等来实现一些类似的功能.Python 有很多自带的模块(标准库)和第三方模块,一个模块可以被其他模 ...

  8. python日志模块_Python之日志处理(logging模块)

    转载自:https://www.cnblogs.com/yyds/p/6901864.html 本节内容 日志相关概念 logging模块简介 使用logging提供的模块级别的函数记录日志 logg ...

  9. python中os模块_Python的武器库11:os模块

    说到编程语言python,有一个著名的格言"余生太短,只用python".如果要分析为什么会存在这么一句格言?python的语法并不简单,有复杂难懂的部分,之所以有这样一句格言,是 ...

最新文章

  1. POJ 2418 Hardwood Species(trie 树)
  2. Python语言学习之打印输出那些事:python输出图表和各种吊炸天的字符串或图画、版权声明(如README.md)等之详细攻略
  3. Git中Add后对部分文件进行取消
  4. java break 在if 中使用_java中使用国密SM4算法详解
  5. i春秋DMZ大型靶场实验(四)Hash基础
  6. mysql error 1594_【MySQL】解决mysql的 1594 错误-阿里云开发者社区
  7. ignite mysql_redis、ignite、mysql memory 内存数据库性能测试对比
  8. ProgressBar与Handler的整合应用
  9. 某公司R2631E以太口通过SDH接新桥的ATM交换机,出现丢包问题的解决方法
  10. ViT (Vision Transformer) ---- Transformer Model(2)
  11. phpstudy所需的vc11和vc14运行库
  12. 世界杯花样营销:争夺32亿人眼球中看到三大趋势
  13. 利用神经网络识别12306验证码——(四)数据增广以及训练集、验证集的划分
  14. python新闻标题分类_机器学习也会“标题党”?这个算法能根据标题判断新闻类别...
  15. 各种欢喜怒笑表情符号,聊天时候用
  16. setdbprefs matlab,matlab数据导入与导出
  17. 【电脑开机没反应的常见原因和解决方法】
  18. SpringBoot-Mongodb双数据源配置
  19. 计算机专业可取证书名称,职业中专计算机维修专业发的职业资格证书名称叫什..._出版资格_帮考网...
  20. 需求定律公式和需求弹性推导——《可以量化的经济学》

热门文章

  1. moocpython123输入若干数、每行输入一个数值作业_MOOC嵩天@python123作业
  2. python的基础字典_python 基础之字典一
  3. 全志a33android编译,编译lichee - 全志A33的环境搭建
  4. netbeans php 安装教程,php_xdebug安装+NetBeans的配置和使用
  5. 在spark应用程序中设置输出日志的级别
  6. c语言中的数据变量类型,大小
  7. Android 内存泄漏分析指北
  8. android -------- Data Binding的使用 ( 四 )ListView
  9. vmware工具克隆linux系统步骤及配置
  10. 解决Ajax中的缓存问题