【LINUX/UNIX网络编程】之简单多线程服务器(多人群聊系统)

RT,Linux下使用c实现的多线程服务器。这个真是简单的不能再简单的了,有写的不好的地方,还希望大神轻拍。(>﹏<)

本学期Linux、unix网络编程的第四个作业。

先上实验要求:

【实验目的】

1、熟练掌握线程的创建与终止方法;

2、熟练掌握线程间通信同步方法;

3、应用套接字函数完成多线程服务器,实现服务器与客户端的信息交互。

【实验内容】

通过一个服务器实现最多5个客户之间的信息群发。

服务器显示客户的登录与退出;

客户连接后首先发送客户名称,之后发送群聊信息;

客户输入bye代表退出,在线客户能显示其他客户的登录于退出。

实现提示:

1、服务器端:

主线程:

定义一个全局客户信息表ent,每个元素对应一个客户,存储:socket描述符、客户名、客户IP、客户端口、状态(初值为0)。

主线程循环接收客户连接请求,在ent中查询状态为0的元素,

如果不存在状态为0的元素(即连接数超过最大连接数),向客户发送EXIT标志;

否则,修改客户信息表中该元素的socket描述符、客户IP、客户端口号,状态为1(表示socket可用);

同时创建一个通信线程并将客户索引号index传递给通信线程。

通信线程:

首先向客户端发送OK标志

循环接收客户发来信息,若信息长度为0,表示客户端已关闭,向所有在线客户发送该用户退出;

若信息为用户名,修改全局客户信息表ent中index客户的用户名name,并显示该用户登录;

若信息为退出,修改全局客户信息表ent中index客户状态为0,并显示该用户退出,终止线程;

同时查询全局客户信息表ent,向状态为1的客户发送接收的信息。

2、客户端:

根据用户从终端输入的服务器IP地址及端口号连接到相应的服务器;

连接成功后,接收服务端发来的信息,若为EXIT,则达到最大用户量,退出;

若为OK,可以通讯,首先先发送客户名称;

主进程循环从终端输入信息,并将信息发送给服务器;

当发送给服务器为bye后,程序退出。

同时创建一个线程负责接收服务器发来的信息,并显示,当接收的长度小于等于0时终止线程;

有了上一次多进程服务器的编写经验以后,写起多线程就简单多了。

照例还是绘制一下流程图,以方便我们理清思路。

好啦,现在可以开始撸代码了。

先实现一下用于通信的结构体clientmsg.h:(和多进程服务器是一样的)

 1 //CLIENTMSG between server and client
 2 #ifndef _clientmsg
 3 #define _clientmsg
 4
 5 //USER MSG EXIT for OP of CLIENTMSG
 6 #define EXIT -1
 7 #define USER 1
 8 #define MSG 2
 9 #define OK 3
10
11 #ifndef CMSGLEN
12 #define CMSGLEN 100
13 #endif
14
15 struct CLIENTMSG{
16     int OP;
17     char username[20];
18     char buf[CMSGLEN];
19 };
20
21 #endif

客户端程序看起来比较简单,咱们把它实现了,client.c:

  1 #include <stdio.h>
  2 #include <string.h>
  3 #include <sys/socket.h>
  4 #include <netinet/in.h>
  5 #include <stdlib.h>
  6 #include <sys/types.h>
  7 #include <sys/wait.h>
  8 #include <signal.h>
  9 #include <unistd.h>
 10 #include <pthread.h>
 11 #include "clientmsg.h"
 12
 13 struct ARG{
 14     int sockfd;
 15     struct CLIENTMSG clientMsg;
 16 };
 17
 18 void *func(void *arg);
 19 void process_cli(int sockfd,struct CLIENTMSG clientMsg);
 20 int main(){
 21     int sockfd;
 22     char ip[20];
 23     int port;
 24     pthread_t tid;
 25     struct sockaddr_in server;
 26     struct CLIENTMSG clientMsgSend;
 27     struct ARG *arg;
 28     /*---------------------socket---------------------*/
 29     if((sockfd = socket(AF_INET,SOCK_STREAM,0))== -1){
 30         perror("socket error\n");
 31         exit(1);
 32     }
 33
 34     /*---------------------connect--------------------*/
 35     printf("Please input the ip:\n");
 36     scanf("%s",ip);
 37     printf("Please input the port:\n");
 38     scanf("%d",&port);
 39     bzero(&server,sizeof(server));
 40     server.sin_family = AF_INET;
 41     server.sin_port = htons(port);
 42     inet_aton(ip,&server.sin_addr);
 43     if(connect(sockfd,(struct sockaddr *)&server,sizeof(server))== -1){
 44         perror("connect() error\n");
 45         exit(1);
 46     }
 47     recv(sockfd,&clientMsgSend,sizeof(clientMsgSend),0);
 48     if(clientMsgSend.OP == OK){
 49         //创建一个线程
 50         arg = (struct ARG *)malloc(sizeof(struct ARG));
 51         arg->sockfd = sockfd;
 52         pthread_create(&tid,NULL,func,(void *)arg);
 53         //主线程
 54         printf("Please input the username:\n");
 55         scanf("%s",clientMsgSend.username);
 56         clientMsgSend.OP = USER;
 57         send(sockfd,&clientMsgSend,sizeof(clientMsgSend),0);
 58         while(1){
 59             clientMsgSend.OP = MSG;
 60             scanf("%s",clientMsgSend.buf);
 61             if(strcmp("bye",clientMsgSend.buf) == 0){
 62                 clientMsgSend.OP = EXIT;
 63                 send(sockfd,&clientMsgSend,sizeof(clientMsgSend),0);
 64                 break;
 65             }
 66             send(sockfd,&clientMsgSend,sizeof(clientMsgSend),0);
 67         }
 68         pthread_cancel(tid);
 69     }
 70     else{
 71         printf("以达到最大连接数!\n");
 72     }
 73     /*------------------------close--------------------------*/
 74     close(sockfd);
 75
 76     return 0;
 77 }
 78
 79
 80 void *func(void *arg){
 81     struct ARG  *info;
 82     info = (struct ARG *)arg;
 83     process_cli(info->sockfd,info->clientMsg);
 84     free(arg);
 85     pthread_exit(NULL);
 86 }
 87 void process_cli(int sockfd,struct CLIENTMSG clientMsg){
 88     int len;
 89     while(1){
 90                 bzero(&clientMsg,sizeof(clientMsg));
 91                 len =recv(sockfd,&clientMsg,sizeof(clientMsg),0);
 92                 if(len > 0){
 93                     if(clientMsg.OP ==USER){
 94                         printf("the user %s is login.\n",clientMsg.username );
 95                     }
 96                     else if(clientMsg.OP == EXIT){
 97                         printf("the user %s is logout.\n",clientMsg.username);
 98                     }
 99                     else if(clientMsg.OP == MSG){
100                         printf("%s: %s\n",clientMsg.username,clientMsg.buf );
101                     }
102                 }
103     }
104 }

然后是服务器端的实现,server.c:

  1 #include <stdio.h>
  2 #include <string.h>
  3 #include <sys/socket.h>
  4 #include <netinet/in.h>
  5 #include <stdlib.h>
  6 #include <sys/types.h>
  7 #include <sys/wait.h>
  8 #include <sys/stat.h>
  9 #include <unistd.h>
 10 #include <fcntl.h>
 11 #include <sys/ipc.h>
 12 #include <pthread.h>
 13 #include "clientmsg.h"
 14
 15 struct Entity{
 16     int sockfd;
 17     char username[20];
 18     char buf[CMSGLEN];
 19     struct sockaddr_in client;
 20     int stat;
 21 };
 22
 23 void *func(void *arg);
 24 void communicate_process(int index);
 25 struct Entity ent[5];
 26
 27 int main(){
 28
 29     struct sockaddr_in server;
 30     struct sockaddr_in client;
 31     int listenfd,connetfd;
 32     char ip[20];
 33     int port;
 34     int addrlen;
 35     struct CLIENTMSG clientMsg;
 36     pthread_t tid;
 37     int *arg;
 38     /*---------------------socket-------------------*/
 39     if((listenfd = socket(AF_INET,SOCK_STREAM,0))== -1){
 40         perror("socket() error\n");
 41         exit(1);
 42     }
 43
 44     /*----------------------IO-----------------------*/
 45     printf("Please input the ip:\n");
 46     scanf("%s",ip);
 47     printf("Please input the port:\n");
 48     scanf("%d",&port);
 49
 50     /*---------------------bind----------------------*/
 51     bzero(&server,sizeof(server));
 52     server.sin_family = AF_INET;
 53     server.sin_port = htons(port);
 54     server.sin_addr.s_addr = inet_addr(ip);
 55     if(bind(listenfd,(struct sockaddr *)&server,sizeof(server))== -1){
 56         perror("bind() error\n");
 57         exit(1);
 58     }
 59
 60     /*----------------------listen-------------------*/
 61     if (listen(listenfd,5)== -1){
 62         perror("listen() error\n");
 63         exit(1);
 64     }
 65     int i;
 66     for(i=0;i<5;i++){
 67         ent[i].stat = 0;
 68     }
 69     while(1){
 70         addrlen = sizeof(client);
 71         if((connetfd = accept(listenfd,(struct sockaddr *)&client,&addrlen))== -1){
 72             perror("accept() error\n");
 73             exit(1);
 74         }
 75         int index = 5;
 76         for(i=0;i<5;i++){
 77             if(ent[i].stat == 0){
 78                 index = i;
 79                 break;
 80             }
 81         }
 82         if(index <= 4){
 83             printf("connetfd:%d\n",connetfd );
 84             ent[index].client = client;
 85             ent[index].sockfd = connetfd;
 86             ent[index].stat = 1;
 87             arg = malloc(sizeof(int));
 88             *arg = index;
 89             pthread_create(&tid,NULL,func,(void *)arg);
 90
 91         }
 92         else{
 93             bzero(&clientMsg,sizeof(clientMsg));
 94             clientMsg.OP = EXIT;
 95             send(connetfd,&clientMsg,sizeof(clientMsg),0);
 96             close(connetfd);
 97         }
 98
 99     }
100
101     /*----------------------close-------------------*/
102
103     close(listenfd);
104
105     return 0;
106 }
107
108
109 /*----------------------------函数实现区----------------------------*/
110 void *func(void *arg){
111     int *info ;
112     info = (int *)arg;
113     communicate_process( *info);
114     pthread_exit(NULL);
115 }
116 void communicate_process(int index){
117     struct CLIENTMSG sendMsg;
118     struct CLIENTMSG recvMsg;
119     printf("sockfd:%d\n",ent[index].sockfd );
120     sendMsg.OP = OK;
121     send(ent[index].sockfd,&sendMsg,sizeof(sendMsg),0);
122
123     while(1){
124         bzero(&sendMsg,sizeof(sendMsg));
125         bzero(&recvMsg,sizeof(recvMsg));
126         int len =recv(ent[index].sockfd,&recvMsg,sizeof(recvMsg),0);
127         if(len > 0){
128             if(recvMsg.OP == USER){
129                 printf("user %s login from ip:%s,port:%d\n",recvMsg.username,inet_ntoa(ent[index].client.sin_addr),ntohs(ent[index].client.sin_port) );
130                 bcopy(recvMsg.username,ent[index].username,strlen(recvMsg.username));
131                 sendMsg.OP = USER;
132             }
133             else if(recvMsg.OP == EXIT){
134                 printf("user %s is logout\n",recvMsg.username );
135                 sendMsg.OP = EXIT;
136                 ent[index].stat = 0;
137                 int i;
138                 for(i=0;i<5;i++){
139                      if(ent[i].stat == 1){
140
141                          send(ent[i].sockfd,&sendMsg,sizeof(sendMsg),0);
142                      }
143                  }
144                 break;
145             }
146             else if(recvMsg.OP == MSG){
147                 sendMsg.OP = MSG;
148             }
149             bcopy(recvMsg.username,sendMsg.username,strlen(recvMsg.username));
150             bcopy(recvMsg.buf,sendMsg.buf,strlen(recvMsg.buf));
151             int i;
152             for(i=0;i<5;i++){
153                  if(ent[i].stat == 1){
154                      printf("stat 1...\n");
155                      send(ent[i].sockfd,&sendMsg,sizeof(sendMsg),0);
156                  }
157              }
158         }
159         else{
160             continue;
161         }
162     }
163 }

最后是makefile文件:

main:server.o client.o
    gcc server.o -oserver -lpthread
    gcc client.o -oclient -lpthread
server.o:server.c clientmsg.h
    gcc -c server.c
client.o:client.c clientmsg.h
    gcc -c client.c

如果程序中引入了#include <pthread.h>,要记得在编译的时候 加上 -lpthread。

下面上一下演示过程:(测试环境,Red Hat Enterprise Linux 6 + centos系Linux,ubuntu下可能会有些问题。)

首先先把服务端启动开来,为了方便测试,这里直接使用的是127.0.0.1的localhost。

然后启动两个客户端用来测试,在用户登录的时候客户端会有消息提醒。服务端会有日志打印输出客户端的名字和登录ip、端口。

客户可以发送消息了,如图发送与接收均正常。这里为了简单演示只是启动了2个。

输入bye以后即可退出聊天并下线。当有客户下线的时候,在线的客户端会收到下线提醒,客户端会有日志打印输出。

时间: 2024-10-23 19:43:06

【LINUX/UNIX网络编程】之简单多线程服务器(多人群聊系统)的相关文章

UNIX网络编程入门——TCP客户/服务器程序详解

前言 最近刚开始看APUE和UNP来学习socket套接字编程,因为网络这方面我还没接触过,要等到下学期才上计算机网络这门课,所以我就找了本教材啃了一两天,也算是入了个门. 至于APUE和UNP这两本书,书是好书,网上也说这书是给进入unix网络编程领域初学者的圣经,这个不可置否,但这个初学者,我认为指的是接受过完整计算机本科教育的研究生初学者,需要具有完整计算机系统,体系结构,网络基础知识.基础没打好就上来啃书反而会适得其反,不过对于我来说也没什么关系,因为基础课也都上得差不多了,而且如果书读

《UNIX网络编程》入门客户端服务器例子

最近在看<UNIX网络编程>(简称unp)和<Linux程序设计>,对于unp中第一个获取服务器时间的例子,实践起来总是有点头痛的,因为作者将声明全部包含在了unp.h里,导致后面编写代码会对这个头文件造成依赖,而学习不到调用了相应功能之后,应该包含哪些确切的头文件. 再者,我下载了unp.h之后,头文件包含再次产生了其他的依赖缺失,因此便参考了<Linux程序设计>中socket一章的入门例子的头文件包含,并且编译中仍然找不到的包含或者是宏定义在unp.h中搜索并粘贴

unix网络编程各种TCP客户-服务器程序设计实例附环境搭建和编译方法(一)

一,到http://download.csdn.net/detail/ts173383201/4505201去下载源代码,然后解压: 二,cd到你解压后的文件夹下,就是有configure的那个目录下,执行命令./configure: 三,执行cd lib跳到lib目录下,执行make命令,会在上层目录(就是刚才有configure那个目录)生成libunp.a文件 四,复制这个静态库libunp.a到/usr/lib/和/usr/lib64/中; 五,接下来在目录中找到unp.h和config

【UNIX网络编程】TCP客户/服务器程序示例

做一个简单的回射服务器: 客户从标准输入读入一行文本,写给服务器 -> 服务器从网络输入读入这行文本,并回射给客户 -> 客户从网络输入读入这行回射文本,并显示在标准输出上 以下是我的代码(部分.h文件是由unpv13e文件夹中的.c文件改名得到) #include "../unpv13e/unp.h" #include "../unpv13e/apueerror.h" #include "../unpv13e/wrapsock.h"

unix网络编程各种TCP客户-服务器程序设计实例(二)

本节我们接着介绍另外的几种TCP客户-服务器程序: 第四种:TCP并发服务器,每个客户一个子线程 在我们前面的并发服务器程序例子中可以看出:父进程接受连接,派生子进程,子进程处理与客户的交互. 这种模式的问题: fork()是昂贵的.内存映像要从父进程拷贝到子进程,所有描述字要在子进程中复制等等. fork()子进程后,需要用进程间通信在父子进程之间传递信息. 一个进程中的所有线程共享相同的全局内存,这使得线程很容易共享信息,但是这种简易型也带来了同步问题.一个进程中的所有线程不仅共享全局变量,

unix网络编程各种TCP客户-服务器程序设计实例(三)

第五种  TCP预先派生子进程服务器程序: 对预先派生子进程服务器的最后一种改动就是由父进程调用accept,然后再将所接受的已连接描述字传递给子进程.父进程必须跟踪子进程的忙闲状态,以便给空闲子进程传递新的描述字.为每个子进程维护一个信息结构,用来管理各子进程. 在调用fork之前,先创建一个字节流管道(Unix域的字节流套接口),它是Unix域的字节流套接口.当子进程派生后,父进程关闭一个描述字(sockfd[1]),子进程关闭另一个描述字(sockfd[0]),此外,子进程将流管道的字节所

Unix网络编程-poll模型echo服务器

poll函数和select函数差不多.以下是一个简单的回显服务器 #include <iostream> using namespace std; #include <poll.h> #include <limits.h> #define OPEN_MAX 64 int main() { int i, maxi, listenfd, connfd, sockfd; int nready; ssize_t n; socklen_t clilen; struct pollf

《UNIX网络编程》TCP客户端服务器:并发、消息回显

经过小小改动,把前面基础的例子做出一点修改. 并发服务器,服务器每accept一个请求就fork()一个新的子进程. 编译运行方法同前一篇. /*client_tcp.c*/ #include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <sys/types.h> #include <sys/socket.h> #include <sys/un.h> #incl

UNIX网络编程入门——I/O复用

UNIX网络编程入门--TCP客户/服务器程序详解 UNIX网络编程入门--TCP客户/服务器程序存在问题及解决 在介绍I/O复用之前,我们先来看一个情况:运行我们前面两篇文章里面的服务器和客户端程序,当客户端在等待用户输入一行字符时,服务器崩溃或者关机了.此时虽然服务器TCP会正确地发送FIN给客户端TCP,但客户端阻塞于fget函数,等待从标准输入读入,无法及时地知道服务器已经终止,要等到它得到标准输入发送给服务器时才会返回错误. 要解决这个问题,就需要一种能力,能够同时观察多个I/O条件是