【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:(和多进程服务器是一样的)
//CLIENTMSG between server and client
#ifndef _clientmsg
#define _clientmsg //USER MSG EXIT for OP of CLIENTMSG
#define EXIT -1
#define USER 1
#define MSG 2
#define OK 3 #ifndef CMSGLEN
#define CMSGLEN 100
#endif struct CLIENTMSG{
int OP;
char username[];
char buf[CMSGLEN];
}; #endif
客户端程序看起来比较简单,咱们把它实现了,client.c:
#include <stdio.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h>
#include <unistd.h>
#include <pthread.h>
#include "clientmsg.h" struct ARG{
int sockfd;
struct CLIENTMSG clientMsg;
}; void *func(void *arg);
void process_cli(int sockfd,struct CLIENTMSG clientMsg);
int main(){
int sockfd;
char ip[];
int port;
pthread_t tid;
struct sockaddr_in server;
struct CLIENTMSG clientMsgSend;
struct ARG *arg;
/*---------------------socket---------------------*/
if((sockfd = socket(AF_INET,SOCK_STREAM,))== -){
perror("socket error\n");
exit();
} /*---------------------connect--------------------*/
printf("Please input the ip:\n");
scanf("%s",ip);
printf("Please input the port:\n");
scanf("%d",&port);
bzero(&server,sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(port);
inet_aton(ip,&server.sin_addr);
if(connect(sockfd,(struct sockaddr *)&server,sizeof(server))== -){
perror("connect() error\n");
exit();
}
recv(sockfd,&clientMsgSend,sizeof(clientMsgSend),);
if(clientMsgSend.OP == OK){
//创建一个线程
arg = (struct ARG *)malloc(sizeof(struct ARG));
arg->sockfd = sockfd;
pthread_create(&tid,NULL,func,(void *)arg);
//主线程
printf("Please input the username:\n");
scanf("%s",clientMsgSend.username);
clientMsgSend.OP = USER;
send(sockfd,&clientMsgSend,sizeof(clientMsgSend),);
while(){
clientMsgSend.OP = MSG;
scanf("%s",clientMsgSend.buf);
if(strcmp("bye",clientMsgSend.buf) == ){
clientMsgSend.OP = EXIT;
send(sockfd,&clientMsgSend,sizeof(clientMsgSend),);
break;
}
send(sockfd,&clientMsgSend,sizeof(clientMsgSend),);
}
pthread_cancel(tid);
}
else{
printf("以达到最大连接数!\n");
}
/*------------------------close--------------------------*/
close(sockfd); return ;
} void *func(void *arg){
struct ARG *info;
info = (struct ARG *)arg;
process_cli(info->sockfd,info->clientMsg);
free(arg);
pthread_exit(NULL);
}
void process_cli(int sockfd,struct CLIENTMSG clientMsg){
int len;
while(){
bzero(&clientMsg,sizeof(clientMsg));
len =recv(sockfd,&clientMsg,sizeof(clientMsg),);
if(len > ){
if(clientMsg.OP ==USER){
printf("the user %s is login.\n",clientMsg.username );
}
else if(clientMsg.OP == EXIT){
printf("the user %s is logout.\n",clientMsg.username);
}
else if(clientMsg.OP == MSG){
printf("%s: %s\n",clientMsg.username,clientMsg.buf );
}
}
}
}
然后是服务器端的实现,server.c:
#include <stdio.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/ipc.h>
#include <pthread.h>
#include "clientmsg.h" struct Entity{
int sockfd;
char username[];
char buf[CMSGLEN];
struct sockaddr_in client;
int stat;
}; void *func(void *arg);
void communicate_process(int index);
struct Entity ent[]; int main(){ struct sockaddr_in server;
struct sockaddr_in client;
int listenfd,connetfd;
char ip[];
int port;
int addrlen;
struct CLIENTMSG clientMsg;
pthread_t tid;
int *arg;
/*---------------------socket-------------------*/
if((listenfd = socket(AF_INET,SOCK_STREAM,))== -){
perror("socket() error\n");
exit();
} /*----------------------IO-----------------------*/
printf("Please input the ip:\n");
scanf("%s",ip);
printf("Please input the port:\n");
scanf("%d",&port); /*---------------------bind----------------------*/
bzero(&server,sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(port);
server.sin_addr.s_addr = inet_addr(ip);
if(bind(listenfd,(struct sockaddr *)&server,sizeof(server))== -){
perror("bind() error\n");
exit();
} /*----------------------listen-------------------*/
if (listen(listenfd,)== -){
perror("listen() error\n");
exit();
}
int i;
for(i=;i<;i++){
ent[i].stat = ;
}
while(){
addrlen = sizeof(client);
if((connetfd = accept(listenfd,(struct sockaddr *)&client,&addrlen))== -){
perror("accept() error\n");
exit();
}
int index = ;
for(i=;i<;i++){
if(ent[i].stat == ){
index = i;
break;
}
}
if(index <= ){
printf("connetfd:%d\n",connetfd );
ent[index].client = client;
ent[index].sockfd = connetfd;
ent[index].stat = ;
arg = malloc(sizeof(int));
*arg = index;
pthread_create(&tid,NULL,func,(void *)arg); }
else{
bzero(&clientMsg,sizeof(clientMsg));
clientMsg.OP = EXIT;
send(connetfd,&clientMsg,sizeof(clientMsg),);
close(connetfd);
} } /*----------------------close-------------------*/ close(listenfd); return ;
} /*----------------------------函数实现区----------------------------*/
void *func(void *arg){
int *info ;
info = (int *)arg;
communicate_process( *info);
pthread_exit(NULL);
}
void communicate_process(int index){
struct CLIENTMSG sendMsg;
struct CLIENTMSG recvMsg;
printf("sockfd:%d\n",ent[index].sockfd );
sendMsg.OP = OK;
send(ent[index].sockfd,&sendMsg,sizeof(sendMsg),); while(){
bzero(&sendMsg,sizeof(sendMsg));
bzero(&recvMsg,sizeof(recvMsg));
int len =recv(ent[index].sockfd,&recvMsg,sizeof(recvMsg),);
if(len > ){
if(recvMsg.OP == USER){
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) );
bcopy(recvMsg.username,ent[index].username,strlen(recvMsg.username));
sendMsg.OP = USER;
}
else if(recvMsg.OP == EXIT){
printf("user %s is logout\n",recvMsg.username );
sendMsg.OP = EXIT;
ent[index].stat = ;
int i;
for(i=;i<;i++){
if(ent[i].stat == ){ send(ent[i].sockfd,&sendMsg,sizeof(sendMsg),);
}
}
break;
}
else if(recvMsg.OP == MSG){
sendMsg.OP = MSG;
}
bcopy(recvMsg.username,sendMsg.username,strlen(recvMsg.username));
bcopy(recvMsg.buf,sendMsg.buf,strlen(recvMsg.buf));
int i;
for(i=;i<;i++){
if(ent[i].stat == ){
printf("stat 1...\n");
send(ent[i].sockfd,&sendMsg,sizeof(sendMsg),);
}
}
}
else{
continue;
}
}
}
最后是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以后即可退出聊天并下线。当有客户下线的时候,在线的客户端会收到下线提醒,客户端会有日志打印输出。


【LINUX/UNIX网络编程】之简单多线程服务器(多人群聊系统)的更多相关文章
- 【LINUX/UNIX网络编程】之使用消息队列,信号量和命名管道实现的多进程服务器(多人群聊系统)
RT,使用消息队列,信号量和命名管道实现的多人群聊系统. 本学期Linux.unix网络编程的第三个作业. 先上实验要求: 实验三 多进程服务器 [实验目的] 1.熟练掌握进程的创建与终止方法: 2 ...
- 【Linux/unix网络编程】之使用socket进行TCP编程
实验一 TCP数据发送与接收 [实验目的] 1.熟练掌握套接字函数的使用方法. 2.应用套接字函数完成基本TCP通讯,实现服务器与客户端的信息交互. [实验学时] 4学时 [实验内容] 实现一个服务器 ...
- 《UNIX网络编程》TCP客户端服务器例子
最近在看<UNIX网络编程>(简称unp)和<Linux程序设计>,对于unp中第一个获取服务器时间的例子,实践起来总是有点头痛的,因为作者将声明全部包含在了unp.h里,导致 ...
- 【Unix 网络编程】TCP 客户/服务器简单 Socket 程序
建立一个 TCP 连接时会发生下述情形: 1. 服务器必须准备好接受外来的连接.这通常通过调用 socket.bind 和 listen 这三个函数来完成,我们称之为被动打开. 2. 客户通过调用 c ...
- UNIX网络编程——TCP回射服务器/客户端程序
下面通过最简单的客户端/服务器程序的实例来学习socket API. serv.c 程序的功能是从客户端读取字符然后直接回射回去: #include<stdio.h> #include&l ...
- UNIX网络编程——UDP回射服务器程序(初级版本)以及漏洞分析
该函数提供的是一个迭代服务器,而不是像TCP服务器那样可以提供一个并发服务器.其中没有对fork的调用,因此单个服务器进程就得处理所有客户.一般来说,大多数TCP服务器是并发的,而大多数UDP服务器是 ...
- 【LINUX/UNIX网络编程】之使用SOCKET进行UDP编程
先看任务需求: 实验二 UDP数据发送与接收 [实验目的] 1.熟练掌握套接字函数的使用方法. 2.应用套接字函数完成基本UDP通讯,实现服务器与客户端的文件传送 [实验学时] 4学时 [实验内容] ...
- 《UNIX网络编程》UDP客户端服务器:消息回显
udp写的程序相比tcp简单一些,在socket()与bind()之后,不需要connect(),accept()等步骤,直接简化为了sendto()与recvfrom(). 编译运行同前面的tcp. ...
- 《UNIX网络编程》TCP客户端服务器:并发、消息回显
经过小小改动,把前面基础的例子做出一点修改. 并发服务器,服务器每accept一个请求就fork()一个新的子进程. 编译运行方法同前一篇. /*client_tcp.c*/ #include < ...
随机推荐
- NYOJ926(概率)
题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=926 设最终A获胜的概率为P,则B获胜的概率为1-P: 因此我们只需要考虑A获胜的概率即可 ...
- 20145206《Java程序设计》实验五Java网络编程及安全
20145206<Java程序设计>实验五 Java网络编程及安全 实验内容 1.掌握Socket程序的编写: 2.掌握密码技术的使用: 3.设计安全传输系统. 实验步骤 我和201451 ...
- 安装less
1.下载安装iterm(http://www.iterm2.com/) 2.打开iterm,输入 sudo npm install -g less
- 三、jQuery--jQuery基础--jQuery基础课程--第5章 jQuery 操作DOM元素
1.使用attr()方法控制元素的属性 attr()方法的作用是设置或者返回元素的属性,其中attr(属性名)格式是获取元素属性名的值,attr(属性名,属性值)格式则是设置元素属性名的值. 例如,使 ...
- php上传文件进度条
ps:本文转自脚本之家 Web应用中常需要提供文件上传的功能.典型的场景包括用户头像上传.相册图片上传等.当需要上传的文件比较大的时候,提供一个显示上传进度的进度条就很有必要了. 在PHP 5.4以前 ...
- Linux 标准目录结构
初学Linux,首先需要弄清Linux 标准目录结构 / root --- 启动Linux时使用的一些核心文件.如操作系统内核.引导程序Grub等. home --- 存储普通用户的个人文件 ftp ...
- “init terminating in do_boot” Windows10 Rabbit MQ fails to start
在Windows 10环境下安装rabbitmq-server-3.6.2后,CMD中运行命令:rabbitmq-plugins enable rabbitmq_management 报错: { , ...
- php判断当前的访问是手机还是电脑
<?php function check_wap() { if (isset($_SERVER['HTTP_VIA'])) return true; if (isset($_SERVER['HT ...
- [Tools] Eclipse使用小技巧-持续更新
[背景] 使用之中发现一些eclipse使用的小技巧,记录下来供以后查阅 Eclipse保存preferences,并导入到其他workspaces The Export wizard can b ...
- Codeforces Round#250 D. The Child and Zoo(并差集)
题目链接:http://codeforces.com/problemset/problem/437/D 思路:并差集应用,先对所有的边从大到小排序,然后枚举边的时候,如果某条边的两个顶点不在同一个集合 ...