gec*_*iss 2 c sockets linux descriptor
我正在设置服务器客户端应用程序(聊天),并且遇到了套接字/线程问题。服务器和客户端都运行在Linux上。
我希望服务器能够为多个用户提供服务,为此我想使用pthread.h中的线程中的线程,因此每个登录用户都有自己的线程。
这是服务器: main.c
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <pthread.h>
#include "manazerUctov.h"
int pocetVlakienZOP = 0;
int main(int argc, char *argv[])
{
pthread_t* vlaknaZiadosti =
(pthread_t*)malloc(sizeof(pthread_t)*POCET_UZIVATELOV);
int sockfd, newsockfd;
socklen_t cli_len;
struct sockaddr_in serv_addr, cli_addr;
int n;
// Program
if (argc < 2) {
fprintf(stderr, "usage %s port\n", argv[0]);
return 1;
}
bzero((char*) &serv_addr, sizeof (serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(atoi(argv[1]));
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0) {
perror("Error creating socket\n");
return 1;
}
printf("Socket OK\n");
if (bind(sockfd, (struct sockaddr*) &serv_addr, sizeof (serv_addr)) < 0) {
perror("Error binding socket address\n");
return 2;
}
printf("Bind OK\n");
listen(sockfd, 5);
while (1) {
cli_len = sizeof (cli_addr);
newsockfd = accept(sockfd, (struct sockaddr*) &cli_addr, &cli_len);
if (newsockfd < 0) {
perror("ERROR on accept\n");
return 3;
}
printf("Accept OK\n");
client_data* cdata = (client_data*) malloc(sizeof (client_data));
cdata->socket = newsockfd;
pthread_t vlakno;
vlaknaZiadosti[pocetVlakienZOP++] = vlakno;
pthread_create(&vlakno, NULL, &serveClient, &cdata);
}
close(sockfd);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
manazerUctov.h:
#ifndef MANAZERUCTOV_H
#define MANAZERUCTOV_H
#ifndef POCET_UZIVATELOV
#define POCET_UZIVATELOV 256
#endif
#ifdef __cplusplus
extern "C" {
#endif
typedef struct {
char *buffer;
char* msg;
int socket;
} client_data;
void* serveClient(void* pdata);
#ifdef __cplusplus
}
#endif
#endif /* MANAZERUCTOV_H */
Run Code Online (Sandbox Code Playgroud)
manazerUctov.c这里,读取函数给出错误:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "manazerUctov.h"
void* serveClient(void *pdata) {
char buffer[256];
client_data* client = (client_data*)pdata;
bzero(buffer, 256);
int n = read(client->socket, buffer, 255);
if (n < 0) {
perror("Error reading from socket\n");
return NULL;
}
printf("Read from socket\n");
close(client->socket);
}
Run Code Online (Sandbox Code Playgroud)
当客户端发送套接字时,服务器接受它并创建一个线程,但我收到Error read from socket: bad file detector。并且程序卡住了,甚至不会退出。就我而言,套接字尚未关闭。在我实现线程之前,从套接字读取工作正常。
编辑:原始代码有 500 行长,所以我修剪了它,以前的错误不再存在,尽管另一个错误是pthread_create中的&cdata。删除并解决了修剪后的代码和原始代码中的问题,并且服务器正在工作。
您的pthread_create函数调用不正确。
这将指针传递给client_data 指针,而不是指针所指向的内容:
pthread_create(&my_thread, NULL, &my_function, &cdata);
Run Code Online (Sandbox Code Playgroud)
你想要的是:
pthread_create(&my_thread, NULL, &my_function, cdata);
Run Code Online (Sandbox Code Playgroud)
因此,cdata->socket您将获得指针值的低 32 位作为文件描述符,而不是获取 。
一些附加说明:
因为您有一个循环作用域pthread_t my_thread;,所以它在循环的每次迭代中都会消失,因此主线程将无法对其执行操作pthread_join。
所以,当你的线程函数退出时,它将处于未连接(即僵尸)状态。
此外,在每次迭代中,每次pthread_create调用都可能指向my_thread相同的地址。我不确定这会对其他线程原语产生什么影响,但我会保持警惕。
您可能想将 添加pthread_t mythread;到分配的结构中。这样,就保证了它有一个唯一的地址。并且,主线程可以跟踪它已分配的结构。
处理此问题的另一种方法是让您的线程函数pthread_detach在顶部执行 a [或在pthread_create调用中添加适当的属性]
这可能更容易清洁。
另外,请确保您的线程函数关闭其套接字并free在退出之前对结构指针执行 a 操作。