2

我编写了一个代码来打印字符串:使用 pthread 的“线程 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);
}

但结果是:

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。

案例二:

当我更改为新代码时:

#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 次!为什么?

4

2 回答 2

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只输入一个字符,然后尝试向其写入字符串。它应该很容易修复。

于 2012-09-10T11:25:29.703 回答
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函数返回时会消失。

于 2012-09-10T11:27:55.137 回答