Как вы определяете объем системной оперативной памяти Linux в C ++?
Вопрос
Я только что написал следующую функцию C ++, чтобы программно определить, сколько оперативной памяти установлено в системе.Это работает, но мне кажется, что должен быть более простой способ сделать это.Может кто-нибудь сказать мне, если я чего-то не понимаю?
getRAM()
{
FILE* stream = popen( "head -n1 /proc/meminfo", "r" );
std::ostringstream output;
int bufsize = 128;
while( !feof( stream ) && !ferror( stream ))
{
char buf[bufsize];
int bytesRead = fread( buf, 1, bufsize, stream );
output.write( buf, bytesRead );
}
std::string result = output.str();
std::string label, ram;
std::istringstream iss(result);
iss >> label;
iss >> ram;
return ram;
}
Во-первых, я использую popen("head -n1 /proc/meminfo")
чтобы получить первую строку файла meminfo из системы.Вывод этой команды выглядит следующим образом
Общий итог:775280 кБ
Как только я получу этот результат в istringstream
, его просто маркировать, чтобы получить нужную мне информацию.Мой вопрос в том, есть ли более простой способ прочитать выходные данные этой команды?Существует ли стандартный вызов библиотеки C ++ для чтения в объеме системной оперативной памяти?
Решение
В Linux вы можете использовать функцию sysinfo
который задает значения в следующей структуре:
#include <sys/sysinfo.h>
int sysinfo(struct sysinfo *info);
struct sysinfo {
long uptime; /* Seconds since boot */
unsigned long loads[3]; /* 1, 5, and 15 minute load averages */
unsigned long totalram; /* Total usable main memory size */
unsigned long freeram; /* Available memory size */
unsigned long sharedram; /* Amount of shared memory */
unsigned long bufferram; /* Memory used by buffers */
unsigned long totalswap; /* Total swap space size */
unsigned long freeswap; /* swap space still available */
unsigned short procs; /* Number of current processes */
unsigned long totalhigh; /* Total high memory size */
unsigned long freehigh; /* Available high memory size */
unsigned int mem_unit; /* Memory unit size in bytes */
char _f[20-2*sizeof(long)-sizeof(int)]; /* Padding for libc5 */
};
Если вы хотите сделать это исключительно с помощью функций C ++ (я бы придерживался sysinfo
), я рекомендую использовать подход C ++, используя std::ifstream
и std::string
:
unsigned long get_mem_total() {
std::string token;
std::ifstream file("/proc/meminfo");
while(file >> token) {
if(token == "MemTotal:") {
unsigned long mem;
if(file >> mem) {
return mem;
} else {
return 0;
}
}
// ignore rest of the line
file.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
}
return 0; // nothing found
}
Другие советы
Нет необходимости использовать popen()
, вы можете просто прочитать файл самостоятельно.Кроме того, если в первой строке будет не то, что вы ищете, вы потерпите неудачу, так как head -n1
считывает только первую строку, а затем завершает работу.Я не уверен, почему вы так смешиваете ввод-вывод C и C ++;это совершенно нормально, но вам, вероятно, следует выбрать весь C или весь C ++.Я бы, наверное, сделал это примерно так:
int GetRamInKB(void)
{
FILE *meminfo = fopen("/proc/meminfo", "r");
if(meminfo == NULL)
... // handle error
char line[256];
while(fgets(line, sizeof(line), meminfo))
{
int ram;
if(sscanf(line, "MemTotal: %d kB", &ram) == 1)
{
fclose(meminfo);
return ram;
}
}
// If we got here, then we couldn't find the proper line in the meminfo file:
// do something appropriate like return an error code, throw an exception, etc.
fclose(meminfo);
return -1;
}
Помните, что /proc/meminfo - это всего лишь файл.Откройте файл, прочитайте первую строку, закройте файл.Вуаля!
Даже top
(из procps
) разбирает /proc/meminfo
, видеть здесь.