多线程并发服务器端的实现
#include <stdio.h>#include <stdlib.h>#include <string.h>#include <unistd.h>#include <arpa/inet.h>#include <sys/socket.h>#include <pthread.h>#define BUF_SIZE 100#define MAX_CLNT 256void* handle_clnt(void *arg);void send_msg(char *msg, int len);void error_handling(char *msg);pthread_mutex_t mutx;int clnt_socks[MAX_CLNT];int clnt_cnt = 0;int main(int argc, char *argv[]) {// 1.检查输入if (argc != 2) {printf("Usage: %s <port>\n", argv[0]);exit(1);}// 2.初始化套接字int serv_sock;struct sockaddr_in serv_adr;pthread_mutex_init(&mutx, NULL);serv_sock = socket(PF_INET, SOCK_STREAM, 0);memset(&serv_adr, 0, sizeof(serv_adr));serv_adr.sin_family = AF_INET;serv_adr.sin_addr.s_addr = htonl(INADDR_ANY);serv_adr.sin_port = htons(atoi(argv[1]));// 3.bind() 套接字if (bind(serv_sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr)) == -1)error_handling("bind() error");// 4.listen() 监听套接字if (listen(serv_sock, 5) == -1)error_handling("listen() error");struct sockaddr_in clnt_adr;int clnt_adr_sz;int clnt_sock;pthread_t t_id;while (1) {clnt_adr_sz = sizeof(clnt_adr);// 5.接入新的客户端clnt_sock = accept(serv_sock, (struct sockaddr*)&clnt_adr, &clnt_adr_sz);// 6.因为主线程和其他新建的子线程 1 可能同时访问 clnt_socks,所以需要锁pthread_mutex_lock(&mutx); // lockclnt_socks[clnt_cnt++] = clnt_sock; // 主线程要访问临界区,有新的客户连接添加到 clnt_socks 中pthread_mutex_unlock(&mutx); // unlock// 7.创建线程 1 向新接入的客户端提供服务pthread_create(&t_id, NULL, handle_clnt, (void*)&clnt_sock);// 8.调用 pthread_detach 函数从内存中完全销毁已终止的线程pthread_detach(t_id);printf("Connected client IP: %s \n", inet_ntoa(clnt_adr.sin_addr));}close (serv_sock);return 0;}void* handle_clnt(void *arg) {int clnt_sock = *((int*)arg);int str_len = 0, i;char msg[BUF_SIZE];while ((str_len = read(clnt_sock, msg, sizeof(msg))) != 0)send_msg(msg, str_len);pthread_mutex_lock(&mutx); // lockfor (i=0; i<clnt_cnt; i++) { // 访问临界区 clnt_cntif (clnt_sock == clnt_socks[i]) { // 访问临界区 clnt_sockswhile (i++ < clnt_cnt-1)clnt_socks[i] = clnt_socks[i+1]; // 访问临界区 clnt_socksbreak;}}clnt_cnt--; // 访问临界区 clnt_cntpthread_mutex_unlock(&mutx); // unlockclose(clnt_sock);return NULL;}// 这里通过循环向所有的客户端发送消息void send_msg(char *msg, int len) {int i;pthread_mutex_lock(&mutx); // lockfor (i=0; i<clnt_cnt; i++) // 访问临界区 clnt_cntwrite(clnt_socks[i], msg, len); // 访问临界区 clnt_sockspthread_mutex_unlock(&mutx); // unlock}void error_handling(char *msg) {fputs(msg, stderr);fputc('\n', stderr);exit(1);}

所以写这些代码的时候,先不管临界区。把其他的代码写完以后,通过变量来划分临界区。
客户端
#include <stdio.h>#include <stdlib.h>#include <string.h>#include <unistd.h>#include <arpa/inet.h>#include <sys/socket.h>#include <pthread.h>#define BUF_SIZE 100#define NAME_SIZE 20void* send_msg(void *arg);void* recv_msg(void *arg);void error_handling(char *msg);char name[NAME_SIZE] = "[DEFAULT]";char msg[BUF_SIZE];int main(int argc, char *argv[]) {// 1.检查输入if (argc != 4) {printf("Usage: %s <IP> <port> <name> \n", argv[0]);exit(1);}sprintf(name, "[%s]", argv[3]);// 2.初始化套接字int sock;struct sockaddr_in serv_addr;sock = socket(PF_INET, SOCK_STREAM, 0);memset(&serv_addr, 0, sizeof(serv_addr));serv_addr.sin_family = AF_INET;serv_addr.sin_addr.s_addr = inet_addr(argv[1]);serv_addr.sin_port = htons(atoi(argv[2]));// 3.向服务器端发起请求 connectif (connect(sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) == -1)error_handling("connect() error");// 4.创建线程,一个线程用于发送消息,另一个线程用于接收消息pthread_t snd_thread, rcv_thread;pthread_create(&snd_thread, NULL, send_msg, (void*)&sock);pthread_create(&rcv_thread, NULL, recv_msg, (void*)&sock);// 5.使用 thread_join 来监视线程的结束void *thread_return;pthread_join(snd_thread, &thread_return);pthread_join(rcv_thread, &thread_return);close(sock);return 0;}void* send_msg(void *arg) {int sock = *((int*) arg);char name_msg[NAME_SIZE+BUF_SIZE];while(1) {fgets(msg, BUF_SIZE, stdin);if (!strcmp(msg, "q\n") || !strcmp(msg, "Q\n")) {close (sock);exit(0);}sprintf(name_msg, "%s %s", name, msg);write(sock, name_msg, strlen(name_msg));}return NULL;}void* recv_msg(void *arg) {int sock = *((int*)arg);char name_msg[NAME_SIZE+BUF_SIZE];int str_len;while(1) {str_len = read(sock, name_msg, NAME_SIZE+BUF_SIZE-1);if (str_len == -1)return (void*)-1;name_msg[str_len] = 0;fputs(name_msg, stdout);}return NULL;}void error_handling(char *msg) {fputs(msg, stderr);fputc('\n', stderr);exit(1);}
运行以上程序
客户端:

服务器端:

