C / POSIX
Este programa usa o número de links físicos para o seu próprio executável como contador da frequência com que foi chamado. Ele cria os novos links físicos no diretório em que foi iniciado (porque dessa forma é garantido que ele esteja no mesmo sistema de arquivos), que, portanto, precisa de permissão de gravação. Omiti o tratamento de erros.
É melhor garantir que você não tenha um arquivo importante com o mesmo nome que um dos links físicos criados nesse diretório, ou ele será substituído. Se, por exemplo, o nome do executável counter
, os links físicos serão nomeados counter_1
, counter_2
etc.
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char* argv[])
{
/* get persistent counter */
struct stat selfstat;
stat(argv[0], &selfstat);
int counter = selfstat.st_nlink;
/* determine digits of counter */
int countercopy = counter;
int digits = 1;
while (countercopy /= 10)
++digits;
/* increment persistent counter */
char* newname = malloc(strlen(argv[0]) + digits + 2);
sprintf(newname, "%s_%d", argv[0], counter);
link(argv[0], newname);
/* output the counter */
if (counter & (counter-1)) // this is zero iff counter is a power of two
printf("%d\n", counter);
else
{
/* determine which power of 2 it is */
int power = 0;
while (counter/=2)
++power;
printf("2^%d\n", power);
}
return 0;
}
Exemplo de execução (a primeira linha redefine o contador, caso o executável já tenha sido executado):
$ rm counter_*
$ ./counter
2^0
$ ./counter
2^1
$ ./counter
3
$ ./counter
2^2
$ ./counter
5
$ ./counter
6
$ ./counter
7
$ ./counter
2^3
$ ./counter
9
$ ls counter*
counter counter_2 counter_4 counter_6 counter_8 counter.c
counter_1 counter_3 counter_5 counter_7 counter_9 counter.c~
0
na primeira execução?