基于Linux下的多进程多线程的聊天室: 服务器:1个 客户端:2~3个(演示时2~3个,实际是可以支持多用户连接登录的)可群聊与私聊的C语言代码

时间: 2024-05-07 16:20:04 浏览: 15
以下是一个基于Linux下的多进程多线程的聊天室的简单实现。代码使用C语言编写。 服务器代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <pthread.h> #include <sys/socket.h> #include <netinet/in.h> #define MAX_CLIENTS 10 #define MAX_MSG_LEN 256 int client_count = 0; int client_fds[MAX_CLIENTS]; pthread_t threads[MAX_CLIENTS]; void *handle_client(void *arg) { int client_fd = *(int *)arg; int i, len; char msg[MAX_MSG_LEN + 1]; while ((len = read(client_fd, msg, MAX_MSG_LEN)) > 0) { msg[len] = '\0'; if (strcmp(msg, "quit") == 0) { break; } for (i = 0; i < client_count; i++) { if (client_fds[i] != client_fd) { write(client_fds[i], msg, strlen(msg)); } } } for (i = 0; i < client_count; i++) { if (client_fds[i] == client_fd) { client_fds[i] = -1; break; } } client_count--; close(client_fd); pthread_exit(NULL); } int main(int argc, char *argv[]) { int server_fd, client_fd, port, i; struct sockaddr_in server_addr, client_addr; socklen_t client_len; if (argc != 2) { fprintf(stderr, "Usage: %s port\n", argv[0]); exit(EXIT_FAILURE); } port = atoi(argv[1]); server_fd = socket(AF_INET, SOCK_STREAM, 0); if (server_fd == -1) { perror("socket"); exit(EXIT_FAILURE); } memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(port); server_addr.sin_addr.s_addr = INADDR_ANY; if (bind(server_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) { perror("bind"); exit(EXIT_FAILURE); } if (listen(server_fd, MAX_CLIENTS) == -1) { perror("listen"); exit(EXIT_FAILURE); } printf("Server started on port %d...\n", port); while (1) { client_len = sizeof(client_addr); client_fd = accept(server_fd, (struct sockaddr *)&client_addr, &client_len); if (client_fd == -1) { perror("accept"); continue; } if (client_count == MAX_CLIENTS) { printf("Max client count reached, closing new connection...\n"); close(client_fd); continue; } for (i = 0; i < MAX_CLIENTS; i++) { if (client_fds[i] == -1) { client_fds[i] = client_fd; pthread_create(&threads[i], NULL, handle_client, (void *)&client_fd); client_count++; break; } } printf("New client connected, count: %d\n", client_count); } close(server_fd); return 0; } ``` 客户端代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <pthread.h> #include <sys/socket.h> #include <netinet/in.h> #define MAX_MSG_LEN 256 void *handle_input(void *arg) { int server_fd = *(int *)arg; int len; char msg[MAX_MSG_LEN + 1]; while (fgets(msg, MAX_MSG_LEN, stdin)) { len = strlen(msg); if (msg[len - 1] == '\n') { msg[len - 1] = '\0'; } write(server_fd, msg, strlen(msg)); } write(server_fd, "quit", strlen("quit")); close(server_fd); pthread_exit(NULL); } void *handle_output(void *arg) { int server_fd = *(int *)arg; int len; char msg[MAX_MSG_LEN + 1]; while ((len = read(server_fd, msg, MAX_MSG_LEN)) > 0) { msg[len] = '\0'; printf("%s\n", msg); } close(server_fd); pthread_exit(NULL); } int main(int argc, char *argv[]) { int server_fd, port; struct sockaddr_in server_addr; if (argc != 3) { fprintf(stderr, "Usage: %s host port\n", argv[0]); exit(EXIT_FAILURE); } port = atoi(argv[2]); server_fd = socket(AF_INET, SOCK_STREAM, 0); if (server_fd == -1) { perror("socket"); exit(EXIT_FAILURE); } memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(port); if (inet_pton(AF_INET, argv[1], &server_addr.sin_addr) != 1) { perror("inet_pton"); exit(EXIT_FAILURE); } if (connect(server_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) { perror("connect"); exit(EXIT_FAILURE); } pthread_t input_thread, output_thread; pthread_create(&input_thread, NULL, handle_input, (void *)&server_fd); pthread_create(&output_thread, NULL, handle_output, (void *)&server_fd); pthread_join(input_thread, NULL); pthread_join(output_thread, NULL); return 0; } ``` 编译命令: ``` gcc server.c -o server -lpthread gcc client.c -o client -lpthread ``` 运行命令: ``` ./server <port> ./client <host> <port> ``` 注意:在服务器端和客户端中,`<port>` 均为要监听的端口号,`<host>` 均为服务器的IP地址。在客户端中输入消息后按回车发送,输入 `quit` 退出。

相关推荐

最新推荐

recommend-type

Python 多线程+多进程简单使用教程,如何在多进程开多线程

一、Python多进程多线程 关于python多进程多线程的相关基础知识,在我之前的博客有写过,并且就关于python多线程的GIL锁问题,也在我的一篇博客中有相关的解释。 为什么python多线程在面对IO密集型任务的时候会产生...
recommend-type

Python3多进程 multiprocessing 模块实例详解

主要介绍了Python3多进程 multiprocessing 模块,结合实例形式详细分析了Python3多进程 multiprocessing 模块的概念、原理、相关方法使用技巧与注意事项,需要的朋友可以参考下
recommend-type

总结:linux进程间通信的几种机制的比较及适用场合

消息队列和过程调用往往单独使用,也就是说它们通常提供了自己的同步机制.相反,共享内存区通常需要由应用程序提供的某种同步形式才能正常工作.解决某个特定问题应使用哪种IPC不存在简单的判定,应该逐渐熟悉各种IPC...
recommend-type

详解Python中logging日志模块在多进程环境下的使用

许多应用程序中都会有日志模块,用于记录系统在运行过程中的一些关键信息,以便于对系统的运行...这篇文章给大家主要介绍了在Python中logging日志模块在多进程环境下的使用,需要的朋友可以参考借鉴,下面来一起看看吧。
recommend-type

Python多进程写入同一文件的方法

今天小编就为大家分享一篇Python多进程写入同一文件的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

MATLAB柱状图在信号处理中的应用:可视化信号特征和频谱分析

![matlab画柱状图](https://img-blog.csdnimg.cn/3f32348f1c9c4481a6f5931993732f97.png) # 1. MATLAB柱状图概述** MATLAB柱状图是一种图形化工具,用于可视化数据中不同类别或组的分布情况。它通过绘制垂直条形来表示每个类别或组中的数据值。柱状图在信号处理中广泛用于可视化信号特征和进行频谱分析。 柱状图的优点在于其简单易懂,能够直观地展示数据分布。在信号处理中,柱状图可以帮助工程师识别信号中的模式、趋势和异常情况,从而为信号分析和处理提供有价值的见解。 # 2. 柱状图在信号处理中的应用 柱状图在信号处理
recommend-type

解释这行代码 c = ((double)rand() / RAND_MAX) * (a + b - fabs(a - b)) + fabs(a - b);

这行代码是用于生成 a 和 b 之间的随机数。首先,它使用 rand() 函数生成一个 [0,1) 之间的随机小数,然后将这个小数乘以 a、b 范围内的差值,再加上 a 和 b 中的较小值。这可以确保生成的随机数大于等于 a,小于等于 b,而且不会因为 a 和 b 之间的差距过大而导致难以生成足够多的随机数。最后,使用 fabs() 函数来确保计算结果是正数。
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。