pthread中的参数传递错误

时间:2012-09-10 11:19:54

标签: linux multithreading parameters pthreads arguments

我编写了一个代码来打印字符串:使用pthread将“Thread 0”改为“Thread 4”。

这是我的代码:

案例1:

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

void *print_message_function(void* parameter) {
    long *i = (long *)parameter;
    printf("Thread %ld\n", *i);
    pthread_exit(0);
}

int main(int argc, char *argv[]) {
    pthread_t threads[5];
    long i = 0;
    for (i = 0; i < 5; i++) {
        pthread_create(&threads[i], 0, print_message_function, (void *)&i);
    }

    pthread_exit(NULL);
}

但结果是:

Thread 2
Thread 3
Thread 3
Thread 4
Thread 5

或:

Thread 0
Thread 0
Thread 0
Thread 0
Thread 0

当我再次运行时它发生了变化。所以我不知道为什么我传递的值是(2到5)或全部(0)或.....(很多情况)。我想我通过的论点是从0到4。

案例2:

当我更改为新代码时:

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

void *print_message_function(void *parameter);

int main(int argc, char *argv[]) {
    pthread_t threads[5];
    int i = 0;
    for (i = 0; i < 5; i++) {
        char *msg = (char*)malloc(sizeof(char));
        sprintf(msg, "Thread %d", i);
        pthread_create(&threads[i], 0, print_message_function, (void *)msg);
    }
}

void *print_message_function(void *parameter) {
    printf("%s\n", (char *)parameter);
}

结果是:

Thread 1
Thread 0
Thread 3
Thread 2
Thread 4
Thread 4

这意味着循环运行了6次!为什么呢?

2 个答案:

答案 0 :(得分:4)

案例1 更改为:

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

void *print_message_function(void* parameter) {
    long i = (long)parameter;  // <<<
    printf("Thread %ld\n", i); // <<<
    pthread_exit(0);
}

int main(int argc, char *argv[]) {
    pthread_t threads[5];
    long i = 0;
    for (i = 0; i < 5; i++) {
        pthread_create(&threads[i], 0, print_message_function, (void *)i); // <<<
    }

    pthread_exit(NULL);
}

之前看到不一致结果的原因是因为您向每个线程传递一个指针,其中每个指针指向同一个局部变量,然后您正在修改它。

案例2 中,您malloc只有一个字符,然后尝试为其写一个字符串。它应该很容易修复。

答案 1 :(得分:2)

您的案例2 方法有效,但您需要修复malloc部分以分配足够的字节。将其更改为

char *msg = (char*)malloc(sizeof(char) * (strlen("Thread ") + 10));
// assuming i will take at most 9-digits (unlikely case)

对于案例1 ,您传递的是i的地址。但是i的值正在改变线程函数将在尝试打印时获得该位置的任何值。另请注意,i的地址可能在线程函数执行时在堆栈上分配时无效,并且在main函数返回时将消失。

相关问题