我编写了一个代码来打印字符串:使用pthread将“Thread 0”改为“Thread 4”。
这是我的代码:
#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次!为什么呢?
答案 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
函数返回时将消失。