我有一个如下的客户端程序,我需要使其成为多线程,即每个连接一个线程。但变量sockfd将保持全局为一个线程。我明白这样做我需要使用pthread_key_t,pthread_key_create等。但是,我很困惑如何使用它。我会感激任何帮助。
int sockfd;
pthread_key_t key_to_sockfd;
void error(const char *msg)
{
perror(msg);
exit(0);
}
void set_connection(char *argv[])
{
int portno;
struct sockaddr_in serv_addr;
struct hostent *server;
char buffer[256];
portno = atoi(argv[2]);
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
server = gethostbyname(argv[1]);
if (server == NULL) {
fprintf(stderr,"ERROR, no such host\n");
exit(0);
}
bzero((char *) &serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
bcopy((char *)server->h_addr,
(char *)&serv_addr.sin_addr.s_addr,server->h_length);
serv_addr.sin_port = htons(portno);
if (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0)
error("ERROR connecting");
return;
}
void send_message()
{
char buffer[256];
int i=0,n;
do{
printf("Please enter the message: ");
bzero(buffer,256);
fgets(buffer,255,stdin);
n = write(sockfd,buffer,strlen(buffer));
if (n < 0)
error("ERROR writing to socket");
i++;
}while(i<3);
bzero(buffer,256);
n = read(sockfd,buffer,255);
if (n < 0)
error("ERROR reading from socket");
printf("%s\n",buffer);
return;
}
void disconnect()
{
close(sockfd);
return;
}
void client_thrd(char *argv[])
{
set_connection(argv);
send_message();
disconnect();
}
int main(int argc, char *argv[])
{
pthread_t thid[2];
int i;
void *status;
if (argc < 3) {
fprintf(stderr,"usage %s hostname port\n", argv[0]);
exit(0);
}
for(i=0;i<1;i++)
pthread_create(&thid[i],NULL,(void*)&client_thrd,(void*)argv);
for(i=0;i<1;i++)
pthread_join(thid[i],&status);
return 0;
}
我已经为一个线程执行了程序并且工作正常。但是当我将线程的数量增加到多于一个时,由于全局可变的sockfd,它不起作用。这是一个更大的测试程序。
谢谢和问候
答案 0 :(得分:3)
它非常易于使用。您在程序开始时调用pthread_key_create(&key_to_sockfd, NULL)
,然后在每个线程中,您将看到初始值为NULL。在每个线程中使用函数pthread_setspecific(key_to_sockfd, <pointer to a struct which contains sockfd>)
。
从那时起,每个线程都会看到一个指向包含sockfd的结构的不同指针。完成线程后,删除结构并使用pthread_setspecific(key_to_sockfd, NULL)
线程完成后,调用pthread_key_delete(key_to_sockfd)
删除存储。您还可以通过向pthread_key_create
提供回调函数来自动清理,以便在线程完成时释放内存。
答案 1 :(得分:2)
从我看到你不需要一个全局变量来实现你想要做的事情。创建“状态”数据结构并在启动之前为每个线程初始化它。线程接口中的void*
参数是为此做的。
答案 2 :(得分:0)
如果我查看你的代码,你用每个线程覆盖你启动套接字描述符sockfd。因此每个线程将建立一个新连接并获得一个新的sockfd。使用此代码,您很可能一个线程正在关闭另一个线程的连接。如果你想为每个线程使用它自己的套接字描述符,你为什么不使用指针在同一个线程中共享同一个套接字描述符呢?
void client_thrd(char *argv[])
{
int sockfd;
set_connection(&sockfd, argv);
send_message(&sockfd);
disconnect(&sockfd);
}