Pequeña ayudita porfavor (threads/hilos) :)

Iniciado por Thorn14, 9 Enero 2014, 19:15 PM

0 Miembros y 2 Visitantes están viendo este tema.

Thorn14

Hola, buenas a todos!

Tengo un problemilla con este programa.

Tengo que hacer que de como resultado el numero pi pero no se que hago mal que las interaciones no me funcionan correctamente (algunos numeros se repiten) y hace que el proceso falle. Esta hecho con threads. Cualquier ayuda sera bienvenida!

# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <unistd.h>
# include <pthread.h>
double factores [1000];
int i,N;
double step,x, totalsum, pi;

void *calcularsum (void *argument){
x = (i+0.5)*step;
factores[i] = 4.0/(1.0+x*x);
}

int main (int argc, char *argv[]) {
if (argc != 2){
printf("Se ha de introducir un valor. Ejemplo: ./pi 10\n");
}else{
N=atoi(argv[1]);
step = 1.0/N;
pthread_t h[N];

for (i=0; i<N; i++) {
pthread_create(&h[i] , NULL , calcularsum , NULL);
 
}
for (i=0; i<N; i++){
pthread_join(h[i], NULL);
totalsum = factores[i] + totalsum;
printf("%f\n",totalsum);
}
pi = totalsum * step;
printf("%.10f\n", pi);
}
return 0;
}

Al hacer el "printf("%f\n",totalsum); " para comprobar que numeros van haciendo los diferentes threads, me encuentro con que alguna vez (random) la hace bien, pero lo mas normal es que el primer número salga como 0.000000 y/o algunos otros se repitan.

El resultado de ejecucion seria:
./pi 10
0.000000
3.911980
7.676686
7.676686
11.003090
14.074107
16.886058
19.446058
21.768264
23.870761
2.3870760525

En cambio otra vez, es un número mas aproximado a pi:
./pi 10
3.990025
7.902005
11.666711
15.230186
18.556589
21.627606
24.439557
26.999557
29.321763
31.424260
3.1424259850

ivancea96

Pon etiquetas de código GeSHi de C o C++ y comentarios en el código para mayor comprensión.

Thorn14

La cosa es que no se si los pthread_create y pthread_join estan hechos correctamente, no domino mucho de hilos

ivancea96

Aunque nunca usé esos threads, te puedo dar esto: Info.
Y decirte, que pruebes en un ejemplo a parte más sencillo.

amchacon

#4
El problema que veo ahí es:
Código (cpp) [Seleccionar]
x = (i+0.5)*step;
factores[i] = 4.0/(1.0+x*x);


Mientras un hilo está modificando la variable x, mientras otros hilos lo están usando. Si tienes suerte y se sincronizan bien pues te sale, pero como modifiques esa variable x mientras lo está usando el otro hilo... O te sale mal el resultado o en algún caso extremo tu progama podría hacer un abort.

Puedes ponerte un mutex para un acceso sincronizado pero, creo que sería más eficiente evitar variables globales que puedan dar colisión. He aquí mi propuesta:
Código (cpp) [Seleccionar]
#include <stdio.h>
#include <pthread.h>

double factores [1000],step;

void *calcularsum (void *argument)
{
   int i = *((int*) argument);
   double x = (i+0.5)*step;
   factores[i] = 4.0/(1.0+x*x);
   free(argument);
}

int main (int argc, char *argv[])
{
   int* P;
   int N,i;
   double totalsum,pi;

   if (argc != 2)
   {
       printf("Se ha de introducir un valor. Ejemplo: ./pi 10\n");
   }
   else
   {
       N=atoi(argv[1]);
       step = 1.0/N;
       pthread_t h[N];

       for (i=0; i<N; i++)
       {
           P = (int*)malloc(sizeof(int));
           *P = i;
           pthread_create(&h[i] , NULL , calcularsum ,P);

       }
       for (i=0; i<N; i++)
       {
           pthread_join(h[i], NULL);
           totalsum = factores[i] + totalsum;
           printf("%f\n",totalsum);
       }
       pi = totalsum * step;
       printf("%.10f\n", pi);
   }
   return 0;
}


No tengo linux a mano ahora, pero debería funcionar.
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar

Thorn14

Si, modificando lo que has comentado, no pasa lo que comentaba. Muchas gracias, ya había pensado también en que podría ser un problema de variables.

Si no es mucha molestia, me podrías explicar que hace las siguientes lineas que has comentado:


    int i = *((int*) argument);

P = (int*)malloc(sizeof(int));
            *P = i;

amchacon

Citarint i = *((int*) argument);
argument es un puntero void, si quiero guardarlo en un entero primero tengo que hacerle un cast a puntero int* y después obtener el valor del entero con un *.

Para profundizar más, mirate explicaciones de punteros en C. Por cierto, son muy importantes.

CitarP = (int*)malloc(sizeof(int));

malloc reserva memoria de forma dinamica, he aquí una explicación de memoria dinámica:
http://foro.elhacker.net/programacion_cc/duda_memoria_dinamica_en_c-t391783.0.html

Ahí uso el new de C++, pero es lo mismo. Después de reservar la memoria guardo un puntero en P, previamente hay que hacer un cast a (int*) ya que P es un puntero a entero.

Citar*P = i;

P es un puntero, para cambiarle el valor a su contenido tengo que usar el operador *
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar

Thorn14

Todo listo, muchas gracias por su tiempo y sus explicaciones.