C, передать синтаксис AWK в качестве аргумента для execl

Я хочу запустить следующую команду из программы на C, чтобы прочитать использование процессора и памяти системы:

ps aux|awk 'NR > 0 { cpu +=$3; ram+=$4 }; END {print cpu,ram}'

Я пытаюсь передать это execl Команда и после этого читать его вывод:

execl("/bin/ps", "/bin/ps", "aux|awk", "'NR > 0 { cpu +=$3; ram+=$4 }; END {print cpu,ram}'",(char *) 0);

Но в терминале я получаю следующую ошибку:

ОШИБКА: неподдерживаемая опция (синтаксис BSD)

Я хотел бы знать, как правильно передать awk в качестве аргумента execl?

3 ответа

Решение

Вы не можете сделать это здесь таким образом.

Проблема в том, что вы хотите выполнить несколько команд. execl для выполнения одной команды. Ваше утверждение использует синтаксис оболочки (особенно |)

Вам повезет, если объединить все это в одну строку и использовать вызов system(3).

Как и предполагал Уилл, popen() - это то, что вы хотите для захвата вывода для последующего использования внутри вашей программы. Однако, если вы действительно хотите выполнить exec-операцию, вы можете использовать оболочку для выполнения ставок через execl():

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char **argv)
{
    printf("%s: execl returned unexpectedly: %d", argv[0],
            execl("/bin/sh", "/bin/sh", "-c",
            "ps aux | awk 'NR >0 { cpu += $3; ram+=$4}; END {print cpu, ram}'",
            NULL));
    exit(1);
}

Вместо того, чтобы запускать awk и анализировать выходные данные awk, вы можете выполнять фильтрацию и суммирование в C, что часто может быстро стать гораздо более удобным. (Это примерно то же самое для точной команды у вас здесь.)

#include <errno.h>
#include <stdio.h>

void ps_cpumem(FILE* f, double* cpu_total, double* mem_total) {
  for (;;) {
    char buf[2000];
    if (!fgets(buf, sizeof buf, f)) {
      return;
    }
    double cpu, mem;
    if (sscanf(buf, "%*s %*s %lf %lf", &cpu, &mem) == 2) {
      *cpu_total += cpu;
      *mem_total += mem;
    }
  }
}

int main() {
  errno = 0;
  FILE* ps = popen("ps aux", "r");
  if (!ps) {
    if (errno == 0) puts("popen: memory allocation failed");
    else perror("popen");
    return 1;
  }
  double cpu = 0, mem = 0;
  ps_cpumem(ps, &cpu, &mem);
  int rc = pclose(ps);
  if (rc == -1) return 1;
  printf("%%cpu: %5.1f\n" "%%mem: %5.1f\n", cpu, mem);
  return 0;
}

Однако вы можете запустить полную команду через popen, так как она выполняет оболочку:

FILE* output = popen("ps aux | awk 'NR > 0 { cpu +=$3; ram+=$4 }; END {print cpu,ram}'", "r");
// read from output to do with as you like
Другие вопросы по тегам