6

我有一个具有 type 字段的类unordered_map。我在我的应用程序中创建了这个对象的一个​​实例,它被包装在一个shared_ptr. 该对象非常消耗内存,我想在使用完它后立即删除它。但是,重置指针只会释放对象占用的一小部分内存。如何强制程序释放对象占用的所有内存?

以下模拟程序重现了我的问题。打印垃圾的 for 循环只是为了给我足够的时间来观察与top. 析构函数在 之后被调用reset()。此外,紧接着,使用的内存从大约 2 GB 下降到 1.5 GB。

#include <iostream>
#include <memory>
#include <unordered_map>

using namespace std;

struct A {
  ~A() {
    cerr << "Destructor" << endl;
  }

  unordered_map<int, int> index;
};

int main() {
  shared_ptr<A> a = make_shared<A>();
  for (size_t i = 0; i < 50000000; ++i) {
    a->index[2*i] = i + 3;
  }

  // Do some random work.
  for (size_t i = 0; i < 3000000; ++i) {
    cout << "First" << endl;
  }

  a.reset();

  // More random work.
  for (size_t i = 0; i < 3000000; ++i) {
    cout << "Second" << endl;
  }
}

编译器:g++ 4.6.3。

4

2 回答 2

5

GCC 的标准库没有“STL 内存缓存”,在其默认配置(几乎每个人都使用)中std::allocator只调用newand delete,它只调用mallocand freemalloc实现(通常来自系统的 C 库)决定是否将内存返回给操作系统。除非您在没有虚拟内存的嵌入式/受限系统上(或者您已关闭过度提交),否则您可能不想返回它——让库做它想做的事。

操作系统不需要内存,它可以毫无问题地为其他应用程序分配千兆字节的虚拟内存。每当人们认为他们需要返回内存时,通常是因为他们不了解现代操作系统如何处理虚拟内存。

如果您真的想强制 C 库将内存返回给操作系统,C 库可能会提供非标准挂钩来执行此操作,例如,对于 GNU libc,您可以调用malloc_trim(0)以强制将最顶部的空闲内存块返回到操作系统,但这可能会使您的程序在下次需要分配更多内存时变慢,因为它必须从操作系统中取回。有关更多详细信息,请参阅https://stackoverflow.com/a/10945602/981959(以及那里的其他答案)。

于 2013-06-19T20:46:20.080 回答
4

无法保证您的应用程序会将内存释放回操作系统。它仍然可供您的应用程序使用,但操作系统可能不会回收它以供一般使用,直到您的应用程序退出。

于 2013-06-19T19:22:49.733 回答