C ++를 사용하여 런타임에 메모리 사용량을 얻는 방법은 무엇입니까?


90

내 프로그램의 런타임에 mem 사용 VIRT 및 RES를 가져 와서 표시해야합니다.

지금까지 시도한 것 :

getrusage ( http://linux.die.net/man/2/getrusage )

int who = RUSAGE_SELF; 
struct rusage usage; 
int ret; 

ret=getrusage(who,&usage);

cout<<usage.ru_maxrss;

하지만 나는 항상 0을 얻습니다.


3
이것은 시스템에 따라 다릅니다. 시스템이 getrusage를 통한 maxrss보고를 지원하지 않는 것 같습니다. 사용중인 배포판을 알려 주실 수 있습니까?
tvanfosson

답변:


79

Linux에서는 ioctl () 솔루션을 찾지 못했습니다 . 우리의 애플리케이션을 위해 우리는 / proc / pid에있는 파일 읽기를 기반으로 일반 유틸리티 루틴을 코딩했습니다 . 다른 결과를 제공하는 이러한 파일이 많이 있습니다. 다음은 우리가 정한 문제입니다 (질문은 C ++ 태그가 지정되었고 C ++ 구조를 사용하여 I / O를 처리했지만 필요한 경우 C i / o 루틴에 쉽게 적용 할 수 있어야합니다).

#include <unistd.h>
#include <ios>
#include <iostream>
#include <fstream>
#include <string>

//////////////////////////////////////////////////////////////////////////////
//
// process_mem_usage(double &, double &) - takes two doubles by reference,
// attempts to read the system-dependent data for a process' virtual memory
// size and resident set size, and return the results in KB.
//
// On failure, returns 0.0, 0.0

void process_mem_usage(double& vm_usage, double& resident_set)
{
   using std::ios_base;
   using std::ifstream;
   using std::string;

   vm_usage     = 0.0;
   resident_set = 0.0;

   // 'file' stat seems to give the most reliable results
   //
   ifstream stat_stream("/proc/self/stat",ios_base::in);

   // dummy vars for leading entries in stat that we don't care about
   //
   string pid, comm, state, ppid, pgrp, session, tty_nr;
   string tpgid, flags, minflt, cminflt, majflt, cmajflt;
   string utime, stime, cutime, cstime, priority, nice;
   string O, itrealvalue, starttime;

   // the two fields we want
   //
   unsigned long vsize;
   long rss;

   stat_stream >> pid >> comm >> state >> ppid >> pgrp >> session >> tty_nr
               >> tpgid >> flags >> minflt >> cminflt >> majflt >> cmajflt
               >> utime >> stime >> cutime >> cstime >> priority >> nice
               >> O >> itrealvalue >> starttime >> vsize >> rss; // don't care about the rest

   stat_stream.close();

   long page_size_kb = sysconf(_SC_PAGE_SIZE) / 1024; // in case x86-64 is configured to use 2MB pages
   vm_usage     = vsize / 1024.0;
   resident_set = rss * page_size_kb;
}

int main()
{
   using std::cout;
   using std::endl;

   double vm, rss;
   process_mem_usage(vm, rss);
   cout << "VM: " << vm << "; RSS: " << rss << endl;
}

다른 * nix 플랫폼에서 / proc / self / stat 구조에 대한 보증이 있습니까? ... 잘 모르겠지만 그렇다면-좋을 것입니다.
bayda 2008 년

음, 수년 동안 저는 주로 Solaris, HP-UX 및 Linux를 사용했습니다. / proc / self / stat는 Linux-ism 인 것 같습니다. 위 프로그램의 원래 버전은 다르기 때문에 Solaris 용 #if 블록을 가졌습니다.
Don Wakefield

나는 OP가 질문 태그를 기반으로 Linux에만 관심이 있다고 가정합니다. / proc을 읽는 것은 당신이 얻는 것만 큼 좋을 것입니다. 솔라리스에서는 kstat를 통해 모든 종류의 정보를 얻을 수 있습니다.
stsquad 2009

파티에 겨우 10 년 늦었지만 vsize를 1024가 아닌 1024.0으로 나누는 이유를 말씀해 주시겠습니까?
a_river_in_canada nov.

1
re : why 1024.0?-컴파일러에게 double FIRST로 변환 한 다음 나누기를 수행하여 이중 결과를 얻도록 지시합니다. 다른 선택 : vm_usage = vsize / 1024;먼저 나누기를 수행 한 다음 (@DonWakefield와 같이 정밀도를 잃음) double로 변환합니다.
Jesse Chisholm

52

David Robert Nadeau 는 자신의 웹 사이트에서 프로세스 상주 집합 크기 (물리적 메모리 사용)를 얻기 위해 좋은 자체 포함 된 다중 플랫폼 C 함수를 넣었습니다 .

/*
 * Author:  David Robert Nadeau
 * Site:    http://NadeauSoftware.com/
 * License: Creative Commons Attribution 3.0 Unported License
 *          http://creativecommons.org/licenses/by/3.0/deed.en_US
 */

#if defined(_WIN32)
#include <windows.h>
#include <psapi.h>

#elif defined(__unix__) || defined(__unix) || defined(unix) || (defined(__APPLE__) && defined(__MACH__))
#include <unistd.h>
#include <sys/resource.h>

#if defined(__APPLE__) && defined(__MACH__)
#include <mach/mach.h>

#elif (defined(_AIX) || defined(__TOS__AIX__)) || (defined(__sun__) || defined(__sun) || defined(sun) && (defined(__SVR4) || defined(__svr4__)))
#include <fcntl.h>
#include <procfs.h>

#elif defined(__linux__) || defined(__linux) || defined(linux) || defined(__gnu_linux__)
#include <stdio.h>

#endif

#else
#error "Cannot define getPeakRSS( ) or getCurrentRSS( ) for an unknown OS."
#endif





/**
 * Returns the peak (maximum so far) resident set size (physical
 * memory use) measured in bytes, or zero if the value cannot be
 * determined on this OS.
 */
size_t getPeakRSS( )
{
#if defined(_WIN32)
    /* Windows -------------------------------------------------- */
    PROCESS_MEMORY_COUNTERS info;
    GetProcessMemoryInfo( GetCurrentProcess( ), &info, sizeof(info) );
    return (size_t)info.PeakWorkingSetSize;

#elif (defined(_AIX) || defined(__TOS__AIX__)) || (defined(__sun__) || defined(__sun) || defined(sun) && (defined(__SVR4) || defined(__svr4__)))
    /* AIX and Solaris ------------------------------------------ */
    struct psinfo psinfo;
    int fd = -1;
    if ( (fd = open( "/proc/self/psinfo", O_RDONLY )) == -1 )
        return (size_t)0L;      /* Can't open? */
    if ( read( fd, &psinfo, sizeof(psinfo) ) != sizeof(psinfo) )
    {
        close( fd );
        return (size_t)0L;      /* Can't read? */
    }
    close( fd );
    return (size_t)(psinfo.pr_rssize * 1024L);

#elif defined(__unix__) || defined(__unix) || defined(unix) || (defined(__APPLE__) && defined(__MACH__))
    /* BSD, Linux, and OSX -------------------------------------- */
    struct rusage rusage;
    getrusage( RUSAGE_SELF, &rusage );
#if defined(__APPLE__) && defined(__MACH__)
    return (size_t)rusage.ru_maxrss;
#else
    return (size_t)(rusage.ru_maxrss * 1024L);
#endif

#else
    /* Unknown OS ----------------------------------------------- */
    return (size_t)0L;          /* Unsupported. */
#endif
}





/**
 * Returns the current resident set size (physical memory use) measured
 * in bytes, or zero if the value cannot be determined on this OS.
 */
size_t getCurrentRSS( )
{
#if defined(_WIN32)
    /* Windows -------------------------------------------------- */
    PROCESS_MEMORY_COUNTERS info;
    GetProcessMemoryInfo( GetCurrentProcess( ), &info, sizeof(info) );
    return (size_t)info.WorkingSetSize;

#elif defined(__APPLE__) && defined(__MACH__)
    /* OSX ------------------------------------------------------ */
    struct mach_task_basic_info info;
    mach_msg_type_number_t infoCount = MACH_TASK_BASIC_INFO_COUNT;
    if ( task_info( mach_task_self( ), MACH_TASK_BASIC_INFO,
        (task_info_t)&info, &infoCount ) != KERN_SUCCESS )
        return (size_t)0L;      /* Can't access? */
    return (size_t)info.resident_size;

#elif defined(__linux__) || defined(__linux) || defined(linux) || defined(__gnu_linux__)
    /* Linux ---------------------------------------------------- */
    long rss = 0L;
    FILE* fp = NULL;
    if ( (fp = fopen( "/proc/self/statm", "r" )) == NULL )
        return (size_t)0L;      /* Can't open? */
    if ( fscanf( fp, "%*s%ld", &rss ) != 1 )
    {
        fclose( fp );
        return (size_t)0L;      /* Can't read? */
    }
    fclose( fp );
    return (size_t)rss * (size_t)sysconf( _SC_PAGESIZE);

#else
    /* AIX, BSD, Solaris, and Unknown OS ------------------------ */
    return (size_t)0L;          /* Unsupported. */
#endif
}

용법

size_t currentSize = getCurrentRSS( );
size_t peakSize    = getPeakRSS( );

자세한 내용은 웹 사이트를 확인하십시오. 또한 시스템의 물리적 메모리 크기를 가져 오는 기능 도 제공 합니다 .


2
범위에 더 잘 추가 #pragma comment(lib, "psapi.lib")했습니다 #if defined(_WIN32).
Bloodmoon

1
@Bloodmon 누군가가 Windows를 사용하지만 Microsoft 컴파일러는 사용하지 않는 경우 어떻게합니까? 그 pragma는 컴파일러를 실패하게 만듭니다.
Adrian

이 코드는 getrusage의 rusage :: ru_maxrss를 사용하며 OP가 그녀를 위해 작동하지 않는다고보고했습니다.
facetus

22

낡은:

maxrss는 프로세스에 사용할 수있는 최대 메모리를 나타냅니다. 0은 프로세스에 제한이 없음을 의미합니다. 아마도 당신이 원하는 것은 비공유 데이터 사용 ru_idrss입니다.

신규 : 커널이 대부분의 값을 채우지 않기 때문에 위의 내용이 실제로 작동하지 않는 것 같습니다. 작동하는 것은 proc에서 정보를 얻는 것입니다. 하지만 스스로 구문 분석하는 대신 다음과 같이 libproc (procps의 일부)을 사용하는 것이 더 쉽습니다.

// getrusage.c
#include <stdio.h>
#include <proc/readproc.h>

int main() {
  struct proc_t usage;
  look_up_our_self(&usage);
  printf("usage: %lu\n", usage.vsize);
}

" gcc -o getrusage getrusage.c -lproc"로 컴파일


1
두 필드 모두 Linux에서 사용할 수 없다는 점을 제외하고는.
jmanning2k

2
이것은 올바르지 않습니다. maxrss는 사용 가능한 최대 값이 아니라 프로세스의 최대 메모리 사용량입니다. 즉, getrlimit (RLIMIT_DATA, & rl)입니다.
jmanning2k

2
#include <proc/readproc.h>솔루션은 Ubuntu에서 저에게 잘 맞았습니다. 나는 패키지를 설치해야했다 libproc-dev. usage.vm_data내가 필요한 것에 가까운 근사치입니다. 선택한 메모리 통계는 여기에 문서화되어 있습니다. /usr/include/proc/readproc.h내가 시도한 모든 항목은 페이지가 아닌 바이트 단위 인 것 같습니다. 내 프로세스가 4,600 만 페이지를 사용하고 있다고 생각하지 않습니다. 이 솔루션이 Linux에서 작동하지 않는다는 의견은 잘못된 것 같습니다.
Allan Stokes

2
올바른 링커는 다음과 같습니다. -lprocps
Sembiance

1
훌륭하게 작동합니다. 이것은 허용되는 대답이어야합니다!
Pekov


8

Don Wakefield 방법의보다 우아한 방법 :

#include <iostream>
#include <fstream>

using namespace std;

int main(){

    int tSize = 0, resident = 0, share = 0;
    ifstream buffer("/proc/self/statm");
    buffer >> tSize >> resident >> share;
    buffer.close();

    long page_size_kb = sysconf(_SC_PAGE_SIZE) / 1024; // in case x86-64 is configured to use 2MB pages
    double rss = resident * page_size_kb;
    cout << "RSS - " << rss << " kB\n";

    double shared_mem = share * page_size_kb;
    cout << "Shared Memory - " << shared_mem << " kB\n";

    cout << "Private Memory - " << rss - shared_mem << "kB\n";
    return 0;
}

7

올바른 값을 얻는 방법에 대해서는 기존 답변이 더 좋지만 적어도 getrusage가 작동하지 않는 이유를 설명 할 수 있습니다.

남자 2 getrusage :

위의 구조체 [rusage]는 BSD 4.3 Reno에서 가져 왔습니다. Linux에서 모든 필드가 의미있는 것은 아닙니다. 현재 (Linux 2.4, 2.6) ru_utime, ru_stime, ru_minflt, ru_majflt 및 ru_nswap 필드 만 유지됩니다.


3

추가로
시스템 ps 명령을 호출하고 출력에서 ​​메모리 사용량을 얻을 수 있습니다.
또는 / proc / pid에서 정보를 읽습니다 (PIOCPSINFO 구조체 참조).


PIOCPSINFO는 내가 사용한 어떤 Linux에서도 실제로 사용할 수 없습니다. / proc / pid에서 읽는 것은 매우 일반적입니다. 나는 ... 대답에 리눅스에 대한 예제 코드를 게시합니다
돈 웨이크 필드

예 / proc / pid 구조는 * nix 플랫폼마다 다르지만 PIOCPSINFO가 있다면 상관 없습니다. 이 구조가 일부 solaris 버전에서 정의되지 않은 상황이 발생했습니다.이 경우에는 ps 출력을 사용했습니다.
bayda 2008 년

2

시스템에라는 파일이 /proc/self/statm있습니다. proc 파일 시스템은 커널 데이터 구조에 대한 인터페이스를 제공하는 의사 파일 시스템입니다. 이 파일에는 공백으로 구분 된 정수만있는 열에 필요한 정보가 포함되어 있습니다.

열 번호 :

  1. = 총 프로그램 크기 (/ proc / [pid] / status의 VmSize)

  2. = 상주 세트 크기 (/ proc / [pid] / status의 VmRSS)

자세한 내용은 LINK를 참조하십시오 .


1

나는 그것을 위해 다른 방법을 사용하고 있으며 현실적으로 들립니다. 내가하는 일은 getpid () 함수로 프로세스의 PID를 얻은 다음 / proc / pid / stat 파일을 사용하는 것입니다. 나는 통계 파일의 23 번째 열이 vmsize라고 생각합니다 (Don 포스트를보세요). 코드에서 필요할 때마다 파일에서 vmsize를 읽을 수 있습니다. 코드 스 니펫이 메모리를 얼마나 사용할 수 있는지 궁금한 경우 해당 스 니펫 이전에 한 번, 이후에 한 번 해당 파일을 읽고 서로 뺄 수 있습니다.


1

변수가 적은 Don W의 솔루션을 기반으로합니다.

void process_mem_usage(double& vm_usage, double& resident_set)
{
    vm_usage     = 0.0;
    resident_set = 0.0;

    // the two fields we want
    unsigned long vsize;
    long rss;
    {
        std::string ignore;
        std::ifstream ifs("/proc/self/stat", std::ios_base::in);
        ifs >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore
                >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore >> ignore
                >> ignore >> ignore >> vsize >> rss;
    }

    long page_size_kb = sysconf(_SC_PAGE_SIZE) / 1024; // in case x86-64 is configured to use 2MB pages
    vm_usage = vsize / 1024.0;
    resident_set = rss * page_size_kb;
}

0

사용 된 최대 메모리를 측정하기 위해 Linux 앱을 찾고있었습니다. valgrind는 훌륭한 도구이지만 내가 원하는 것보다 더 많은 정보를 제공했습니다.tstime 은 내가 찾을 수있는 최고의 도구 인 것 같았습니다. "고수"메모리 사용량 (RSS 및 가상)을 측정합니다. 이 답변을 참조하십시오 .

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.