对于一个 C++ 程序,是否有可能跟踪程序在某一时刻使用了多少内存?
比如,有一个原型为:
int getEstimatedTotalMemoryUsage();
我想,如果不可能的话,那么一个人就必须退出程序,执行系统调用并从那里检查结果。 如果是这样,有哪些可用于此类目的的工具呢?假设这样的事情是可能的。
编辑:我正在使用linux,有没有可以为您执行此操作的工具?
对于一个 C++ 程序,是否有可能跟踪程序在某一时刻使用了多少内存?
比如,有一个原型为:
int getEstimatedTotalMemoryUsage();
我想,如果不可能的话,那么一个人就必须退出程序,执行系统调用并从那里检查结果。 如果是这样,有哪些可用于此类目的的工具呢?假设这样的事情是可能的。
编辑:我正在使用linux,有没有可以为您执行此操作的工具?
是的 - 使用POSIX getrusage
。从Linux man页面:
概要
#include <sys/time.h> #include <sys/resource.h> int getrusage(int who, struct rusage *usage);
描述
getrusage()
函数返回当前资源使用情况,可以为RUSAGE_SELF
或RUSAGE_CHILDREN
中的一个参数。前者请求当前进程使用的资源,后者请求已终止并已等待其子进程使用的资源。
struct rusage { struct timeval ru_utime; /* user time used */ struct timeval ru_stime; /* system time used */ long ru_maxrss; /* maximum resident set size */ long ru_ixrss; /* integral shared memory size */ long ru_idrss; /* integral unshared data size */ long ru_isrss; /* integral unshared stack size */ long ru_minflt; /* page reclaims */ long ru_majflt; /* page faults */ long ru_nswap; /* swaps */ long ru_inblock; /* block input operations */ long ru_oublock; /* block output operations */ long ru_msgsnd; /* messages sent */ long ru_msgrcv; /* messages received */ long ru_nsignals; /* signals received */ long ru_nvcsw; /* voluntary context switches */ long ru_nivcsw; /* involuntary context switches */ };
struct rusage
:http://www.gnu.org/s/libc/manual/html_node/Resource-Usage.html - chrisaycockgetrusage()
,我甚至没有想到它可能在Linux上没有完全实现。那么,有没有其他信息来源可以提供给OP呢?显然,我应该读整个man页面... - Carl Norum#include <windows.h>
#include <Psapi.h>
// [...]
PROCESS_MEMORY_COUNTERS memCounter;
BOOL result = K32GetProcessMemoryInfo(GetCurrentProcess(), &memCounter, sizeof(memCounter));
std::cout << "WorkingSetSize " << memCounter.WorkingSetSize << std::endl;
以下是返回值的解释:https://learn.microsoft.com/zh-cn/windows/win32/api/psapi/ns-psapi-process_memory_counters
我今天自己需要这个,所以在这里分享测试结果。我相信在任何Unix系统上,调用getmem()将会做到OP所要求的。这段代码用非常通用的C语言编写,它可以在C或C++中运行。
// Calling function must free the returned result.
char* exec(const char* command) {
FILE* fp;
char* line = NULL;
// Following initialization is equivalent to char* result = ""; and just
// initializes result to an empty string, only it works with
// -Werror=write-strings and is so much less clear.
char* result = (char*) calloc(1, 1);
size_t len = 0;
fflush(NULL);
fp = popen(command, "r");
if (fp == NULL) {
printf("Cannot execute command:\n%s\n", command);
return NULL;
}
while(getline(&line, &len, fp) != -1) {
// +1 below to allow room for null terminator.
result = (char*) realloc(result, strlen(result) + strlen(line) + 1);
// +1 below so we copy the final null terminator.
strncpy(result + strlen(result), line, strlen(line) + 1);
free(line);
line = NULL;
}
fflush(fp);
if (pclose(fp) != 0) {
perror("Cannot close stream.\n");
}
return result;
}
int getmem() {
pid_t pid = getpid();
char cmd[64];
snprintf(cmd, 64, "/bin/ps -p %d -o size", pid);
char* result = exec(cmd);
if (!result) {
return 0;
}
// Find first newline.
int pos = 0;
while (result[pos] != '\n') {
pos++;
}
// Remove the final newline.
result[strlen(result) - 1] = '\0';
// Convert to integer.
int size = atoi(result + pos + 1);
free(result);
return size;
}
从技术上讲,我想printf(...)这一行应该是fprintf(stderr, ...),但由于某些特定环境需要记录日志,我会将stderr重定向,而这样编译和测试代码也能正常工作,所以我直接复制粘贴以避免出错。
pid_t getpid(void); // unistd.h
解析/proc/<id>/smaps
如果您不关心内存总量中的共享库,那么可能更简单
调用系统调用ps -p <id> -o %mem