Как измерить использование памяти изнутри программы на С++?
Возможно ли для программы С++ отслеживать, сколько памяти используется программой за один раз?
Например, функция с прототипом:
int getEstimatedTotalMemoryUsage();
Я полагаю, что если это невозможно, тогда вам придется выйти из программы, выполнить системный вызов и проверить результаты оттуда. Если да, то какие инструменты доступны для таких целей? Предполагая, что такое возможно, то есть.
edit: Я использую linux, любые инструменты, которые могут сделать это для вас?
Ответы
Ответ 1
Да - используйте 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 */
};
Ответ 2
Я хотел этого сегодня, сам, поэтому поделился результатами тестирования здесь. Я считаю, что вызов getmem() будет делать то, что запросил OP, в любом окне unix. Написанный в очень общем 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 для определенных условий ведения журнала по конкретным окружениям, и именно так я скомпилировал и протестировал код, поэтому я копирую дословно, чтобы избежать поломки.
Ответ 3
Получите ваш PID: pid_t getpid(void); // unistd.h
Анализ /proc/<id>/smaps
Если вам все равно, что общие библиотеки в общей памяти могут быть проще,
сделать системный вызов ps -p <id> -o %mem