Как измерить использование памяти внутри программы на C ++? - PullRequest
31 голосов
/ 25 февраля 2011

Возможно ли для программы на c ++ отслеживать объем памяти, используемый программой за один раз?

Например, функция с прототипом:

int getEstimatedTotalMemoryUsage();

Полагаю, если это невозможно, тогда придется выйти из программы, сделать системный вызов и проверить результаты оттуда. Если да, какие инструменты доступны для таких целей? Предполагая, что такое возможно, то есть.

edit: я использую linux, какие-нибудь инструменты, которые могут сделать это для вас?

Ответы [ 3 ]

34 голосов
/ 25 февраля 2011

Да - используйте POSIX getrusage.Со страницы руководства Linux :

Синопсис

#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 голосов
/ 14 февраля 2014

Я сам хотел этого сегодня, поэтому делюсь результатами тестирования здесь. Я полагаю, что вызов getmem () сделает то, что попросил OP, на любом unix-сервере. Написанный на очень родном 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 по определенным причинам, связанным с журналированием, и именно так я скомпилировал и протестировал код, поэтому я копирую дословно, чтобы избежать поломки.

0 голосов
/ 01 декабря 2011

Получите свой PID: pid_t getpid(void); // unistd.h

Разбор /proc/<id>/smaps

Если вам не нужны общие библиотеки, это может быть проще

сделать системный вызов ps -p <id> -o %mem

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...