Foro de elhacker.net

Programación => Programación C/C++ => Mensaje iniciado por: Andrea0000 en 28 Diciembre 2021, 06:30 am



Título: Duda con un programa en C
Publicado por: Andrea0000 en 28 Diciembre 2021, 06:30 am
Hola, no se muy bien como puedo especificar en mi codigo que en lugar de tener o crear 5 procesos hijos, se creen tantos procesos hijos como numero de argumentos hallan
Código:
#define _POSIX_SOURCE
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <signal.h>
#include <wait.h>

#define NUM_HIJOS 5//El numero de hijos debe ser tantos como numero de parametros haya
#define NOMBREHIJO "hijo"

void FinalizarProcesos();

void Manejador(int num);
pid_t pids[NUM_HIJOS];

int main(int argc, char *argv[]){
int i, j;
int pid;
/*Tratamiento de la linea de ordenes*/
if(argc != 2){
printf(stderr, "Error en el numero de argumentos\n");
exit(EXIT_FAILURE);
}
/*Tratamiento de la señal*/
if(signal(SIGINT, Manejador)== SIG_ERR){
fprintf(stderr, "Error en la manipulacion de la señal\n");
exit(EXIT_FAILURE);
}
/*Creacion de los procesos hijos*/
for(i=0; i<NUM_HIJOS; i++){
switch(pids[i]=fork()){ //Se guarda el pid en la tabla de procesos
case -1:
fprintf(stderr, "Error en la creacion del proceso hijo\n");
FinalizarProcesos();
break;
case 0:
if(execl(NOMBREHIJO, NOMBREHIJO, argv[1], NULL) == -1){
fprintf(stderr, "Error en la ejecucioon del proceso hijo\n");
exit(EXIT_FAILURE);
}
break;
}

}
/*Espera a la finalizacion de los procesos hijos*/
for(i=0; i<NUM_HIJOS; i++){
pid=wait(NULL);
for(j=0;j<NUM_HIJOS; j++){
if(pid==pids[j]){
printf("[Preoceso padre] el proceso %d ha terminado\n", pid);
pids[j]=0; //0 para indicar que no existe el proceso
}
}
}
printf("[Proceso padre] finaliza\n");
return EXIT_SUCCESS;
}

/*FinalizarProcesos: Termina todos los procesos hijos vivos*/
void FinalizarProcesos(void){
int i;

for(i=0; i<NUM_HIJOS; i++){
if(pids[i] != 0){ //Solo se mata a los procesos hijos vivos
if(kill(pids[i], SIGINT) == -1){
fprintf(stderr, "Error al enviar una señal\n");
}
}
}
}

/*Manejador: Manejador de señal*/
void Manejador(int num){
FinalizarProcesos();
printf("[Proceso padre] finaliza\n");
exit(EXIR_SUCCESS);
}


Título: Re: Duda con un programa en C
Publicado por: @XSStringManolo en 28 Diciembre 2021, 07:32 am
Código
  1. #define NUM_HIJOS argc

Esto?


Título: Re: Duda con un programa en C
Publicado por: Andrea0000 en 28 Diciembre 2021, 08:10 am
Puede ser voy a ponerlo, gracias


Título: Re: Duda con un programa en C
Publicado por: K-YreX en 28 Diciembre 2021, 08:21 am
Citar
Si intentas definir esa constante antes del main() tendrás problemas pues argc no existe en ese punto.
EDIT: Tenía mis dudas y tras probarlo, no da problemas (al menos en mi caso). Funciona correctamente.

Puedes usar directamente los parámetros de la función main():
  • argc -> Contiene el número de argumentos. Si no se pasa nada: argc = 1
  • argv -> Contiene los argumentos. Si no se pasa nada: argv[0] = <nombre_programa>

Prueba este código (con diferentes argumentos) para entenderlo mejor:
Código
  1. #include <stdio.h>
  2.  
  3. int main(int argc, char *argv[]) {
  4.  printf("Numero de argumentos: %d\n\n", argc);
  5.  
  6.  for(int i = 0; i < argc; ++i) {
  7.    printf("Argumento %d: %s\n", i, argv[i]);
  8.  }
  9. }


Título: Re: Duda con un programa en C
Publicado por: @XSStringManolo en 28 Diciembre 2021, 18:32 pm
El define antes del main funciona porque lo cambia el preprocesador. Es decir, no es una variable, si no un remplazo.

Al poner NUM_HIJOS, el preprocesador lo remplaza por la expresión que definiste. Por ejemplo:
Código
  1. #define EJEMPLO argc
  2.  
  3. int main(int argc, char *argv[]) {
  4.  std::cout << EJEMPLO;
  5.  return 0;
  6. }

Es exactamente lo mismo que:
Código
  1. int main(int argc, char *argv[]) {
  2.  std::cout << argc;
  3.  return 0;
  4. }
}



Título: Re: Duda con un programa en C
Publicado por: K-YreX en 29 Diciembre 2021, 23:03 pm
Cierto, cierto.
No me di cuenta en ese momento porque estaba contestando todavía con una legaña en el ojo  ;D

Otro pequeño detalle a tener en cuenta es que un programa ejecutado "sin argumentos", realmente se está ejecutando con un argumento: el nombre del programa. Es decir, si se ejecuta un programa sin argumentos, argc vale 1.
Si no se quiere contar este argumento, se podría hacer algo así:
Código
  1. #define NUM_HIJOS (argc - 1) // IMPORTANTE: Usar parentesis

Aunque personalmente preferiría usar argc directamente en vez del define.
Por lo demás, tema zanjado.