Windows метры памяти c для обнаружения утечки памяти - PullRequest
1 голос
/ 04 августа 2020

У нас есть большой старый код устаревшего сервера, работающий как 64-битная windows служба. У службы есть утечка памяти, которую на данный момент у нас нет ресурсов для исправления.

Поскольку служба устойчива к перезапуску, временное ужасное `` решение '', которое мы хотим, - это обнаруживать, когда память службы превышено, например, 5 ГБ, и выйдите из службы (которая имеет автоматический перезапуск для таких случаев).

Мой вопрос в том, для каких метри c следует использовать go? Используется ли GlobalMemoryStatusEx для получения MEMORYSTATUSEX.ullTotalVirtual- MEMORYSTATUSEX.ullAvailVirtual правильно?

1 Ответ

4 голосов
/ 04 августа 2020

GlobalMemoryStatusEx неверно. Вы не хотите заполнять память устройства, пока не останется 5 ГБ.

Вам потребуется GetProcessMemoryInfo.

BOOL WINAPI GetProcessMemoryInfo(
  __in   HANDLE Process,
  __out  PPROCESS_MEMORY_COUNTERS ppsmemCounters,
  __in   DWORD cb
);

Из примера с использованием GetProcessMemoryInfo:

#include <windows.h>
#include <stdio.h>
#include <psapi.h>

// To ensure correct resolution of symbols, add Psapi.lib to TARGETLIBS
// and compile with -DPSAPI_VERSION=1

void PrintMemoryInfo( DWORD processID )
{
    HANDLE hProcess;
    PROCESS_MEMORY_COUNTERS pmc;

    // Print the process identifier.

    printf( "\nProcess ID: %u\n", processID );

    // Print information about the memory usage of the process.

    hProcess = OpenProcess(  PROCESS_QUERY_INFORMATION |
                                    PROCESS_VM_READ,
                                    FALSE, processID );
    if (NULL == hProcess)
        return;

    if ( GetProcessMemoryInfo( hProcess, &pmc, sizeof(pmc)) )
    {
        printf( "\tWorkingSetSize: 0x%08X\n", pmc.WorkingSetSize );
        printf( "\tPagefileUsage: 0x%08X\n", pmc.PagefileUsage ); 
    }

    CloseHandle( hProcess );
}

int main( void )
{
    // Get the list of process identifiers.

    DWORD aProcesses[1024], cbNeeded, cProcesses;
    unsigned int i;

    if ( !EnumProcesses( aProcesses, sizeof(aProcesses), &cbNeeded ) )
    {
        return 1;
    }

    // Calculate how many process identifiers were returned.

    cProcesses = cbNeeded / sizeof(DWORD);

    // Print the memory usage for each process

    for ( i = 0; i < cProcesses; i++ )
    {
        PrintMemoryInfo( aProcesses[i] );
    }

    return 0;
}

Хотя это не интуитивно, вам нужно прочитать PagefileUsage, что даст вам выделенную память, выделенную вашим процессом. WorkingSetSize ненадежен, потому что, если машине не хватает памяти, ОС запишет все данные в файл подкачки. Это может привести к тому, что WorkingSetSize будет маленьким (например, 100 МБ), но на самом деле вы потеряли уже 20 ГБ памяти. Это приведет к зубчатой ​​диаграмме в потреблении памяти, пока файл подкачки не будет заполнен. Рабочий набор - это только активно используемая память, которая может скрыть утечку памяти в несколько ГБ, если машина находится под нехваткой памяти.

...