- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
在使用开源 Cython 库时,我发现了内存泄漏。泄漏似乎来自类型化的 numpy 数组,当它超出范围时,它不会从内存中释放。声明如下:
cdef np.ndarray[object, ndim=1] my_array = np.empty(my_size, dtype=object)
根据我的理解,垃圾收集器应该像任何其他 numpy 数组一样考虑它,并且 GC 应该在数组超出范围时立即释放它的内存——在这种情况下是在它所在的函数的末尾被宣布。显然这不会发生。
如果首先使用 cython 数组创建数组,然后将其转换为 numpy 数组,则可以使用描述的 callback_free_data 函数 here和 here .但是,在这种情况下,无法到达 my_array
的指针,也无法设置回调。
知道为什么这种声明会导致内存泄漏和/或如何强制释放内存吗?
更新:
我的问题很笼统,我想避免发布代码,因为它有点复杂,但既然有人问了,我们就开始吧:
cdef dijkstra(Graph G, int start_idx, int end_idx):
# Some code
cdef np.ndarray[object, ndim=1] fiboheap_nodes = np.empty([G.num_nodes], dtype=object) # holds all of our FiboHeap Nodes Pointers
Q = FiboHeap()
fiboheap_nodes[start_idx] = Q.insert(0, start_idx)
# Some other code where it could perform operations like:
# Q.decrease_key(fiboheap_nodes[w], vw_distance)
# End of operations
# do we need to cleanup the fiboheap_nodes array here?
return
FiboHeap
是 c 实现的 Cython 包装器。例如,插入函数如下所示:
cimport cfiboheap
from cpython.pycapsule cimport PyCapsule_New, PyCapsule_GetPointer
from python_ref cimport Py_INCREF, Py_DECREF
cdef inline object convert_fibheap_el_to_pycapsule(cfiboheap.fibheap_el* element):
return PyCapsule_New(element, NULL, NULL)
cdef class FiboHeap:
def __cinit__(FiboHeap self):
self.treeptr = cfiboheap.fh_makekeyheap()
if self.treeptr is NULL:
raise MemoryError()
def __dealloc__(FiboHeap self):
if self.treeptr is not NULL:
cfiboheap.fh_deleteheap(self.treeptr)
cpdef object insert(FiboHeap self, double key, object data=None):
Py_INCREF(data)
cdef cfiboheap.fibheap_el* retValue = cfiboheap.fh_insertkey(self.treeptr, key, <void*>data)
if retValue is NULL:
raise MemoryError()
return convert_fibheap_el_to_pycapsule(retValue)
__dealloc__()
函数按预期工作,因此 FiboHeap 在函数 dijkstra(...)
结束时从内存中释放。我的猜测是 fiboheap_nodes 中包含的指针出了问题。有什么猜测吗?
最佳答案
问题(在评论中解决)原来不是 numpy 数组的释放。相反,numpy 数组包含一堆 Fiboheap
对象,这些对象本身包含指向一堆 Python 对象的指针。正是这些对象没有被释放。
当 Fiboheap
中的 Python 对象指针被获取(在 insert
中)时,它们的引用计数会增加以确保它们保持事件状态。然而,当 Fiboheap
被销毁时(在 __dealloc__
中),它持有的 Python 对象的引用计数并没有减少,导致内存泄漏。解决方案是确保在 __dealloc__
期间对所有持有的 Python 对象调用 Py_DECREF
。
可能还有第二个更具挑战性的问题等待出现:Fiboheap
持有的对象本身可能包含对 Fiboheap
的引用,也许是间接的。 Python 使用函数 tp_tranverse
找到这些循环和 tp_clear
来打破它们。 Cython 会为其 cdef
类自动生成一个 tp_traverse
,但是因为它无法知道隐藏在 C Fiboheap
中的 Python 对象指针结构它不会正确处理这些(可能会产生另一个内存泄漏)。
这在现实中可能不太可能发生,因此可能不值得担心,但这是需要注意的事情。 newsgroup post描述了一种在 Cython 中生成自定义 tp_traverse
函数的方法。对于大多数应用程序,这应该不是必需的 - 只有 Cython object
和 PyObject*
的混合才使它在这里成为可能。
关于python - 如何释放类型化的 numpy 数组?设置 callback_free_data 是一个可行的选择吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38251216/
我有一个附加了 View Controller 的 AVAudioPlayer 实例。 @property (nonatomic, retain) AVAudioPlayer *previewAudi
我是java初学者。假设我声明了一个 Account 类型的变量 Account _account = new Account("Thomas"); 然后在其他地方我做了这样的事情: _account
我在我的应用程序中使用了 3 个 UIViewController,现在我想知道当我从另一个应用程序切换到另一个 UIViewController 时释放它们是否是一个好主意。显然,这将是隐藏的,当它
我分配了一个直接缓冲区: ByteBuffer directBuffer = ByteBuffer.allocateDirect(1024); 我读过: Deallocating Direct Buf
场景。我有一个图表,我可以使用右键单击来执行平移。这非常有效。然后我完美地添加了右键菜单。 问题。现在,即使在拖动操作完成后释放鼠标,也会显示右键菜单。 有没有办法在 Java Swing 或 Jav
我使用此代码获取 ABPerson 的姓氏 CFStringRef lastNameRef = ABRecordCopyValue((ABRecordRef)personRecordRef, kABP
目前,我们在基于 C 的嵌入式应用程序中使用 malloc/free Linux 命令进行内存分配/取消分配。我听说这会导致内存碎片,因为内存分配/取消分配会导致堆大小增加/减少,从而导致性能下降。其
当我尝试释放缓冲区时遇到问题。每次我尝试将缓冲区传递给释放方法时,都会发生段错误。 Valgrind 确认段错误位于 BufferDeallocate 方法中。 ==30960== Memcheck,
我想知道何时按下或释放修改后的键(Ctrl 或 Shift)。 基本上,用户可以在按下修改键的情况下执行多次击键,而我不想在它被释放之前执行一个操作(想想 Emacs 和 Ctrl + X + S).
我编写了一个相当大的网络应用程序。它运行良好一段时间,然后慢慢开始运行缓慢,因为 DOM 节点开始爬升到 80,000 - 100,000 左右。 所以我一直在 Chrome 开发工具控制台 (DCT
我知道在像 c 这样的语言中,我需要在分配内存后释放它。 (我来自 Java),对此我有几个问题: 当我在做的时候: int array[30]; (即创建一个大小为 30 个整数的数组)与
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: How to release pointer from boost::shared_ptr? Detach
我有一个可以从多个后台线程访问的类,可能同时访问。我无法复制该类,因为重新创建它的内容(处理或内存方面)可能很昂贵。 也有可能在后台处理仍在继续并访问该属性时替换了此类的属性。 目前我有定期的保留/释
这个问题是对: 的扩展链接-1:Creating an image out of the ios surface and saving it Link-2:Taking Screenshots fro
我有一个实例变量 NSMutableArray* searchResults。 首先,我初始化它: self.searchResults = [[NSMutableArray alloc] init]
如果我在堆上声明一些东西,比如 char *a=new char[1000] 并且主程序停止,如果没有 delete[]<,那么分配的内存会发生什么 调用?它保留在堆上还是自动释放? 最佳答案 就C+
在开发相机应用时,我遇到了一个异常,该异常仅在我切换到其他应用时发生(onPause() 用于我的应用)。 01-15 17:22:15.017: E/AndroidRuntime(14336): F
使用 JDK 1.8 编译时出现 maven 编译器错误 无法执行目标 org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (de
将 BufferedImage 保存到磁盘(以释放内存)的最快方法是什么? 我的 Java 应用程序处理大量图像(每约 300 毫秒将图像加载到内存中)。大多数这些图像都会立即被丢弃 (gc),但每隔
使用 JDK 1.8 编译时出现 maven 编译器错误 未能在项目 DUMMY 上执行目标 org.apache.maven.plugins:maven-compiler-plugin:3.8.1:
我是一名优秀的程序员,十分优秀!