Linux socket多进程服务器框架三
在使用select管理服务器连接的时候:
注意1:select是可中断睡眠函数,需要屏蔽信号
注意2:必须获取select的返回值nread,每次处理完一个事件,nread需要-1
注意3:如果客户端的连接超过连接池的大小,需要关闭客户端连接
注意4:获取最大套接字的方法是每次有客户端连接过来时,在和maxfd比较,这样就不用每次select之前都遍历池,查找最大值
服务器
//serhelp.h #ifndef _vxser
#define _vxser #ifdef __cplusplus
extern "C"
{
#endif /**
* sersocket_init - socket初始化
* @listenfd:文件描述符
* 成功返回0,失败返回错误码
* */
int sersocket_init(int *listenfd); /**
* listen_socket - 绑定端口号,监听套接字
* @listenfd:文件描述符
* @port:绑定的端口号
* 成功返回0,失败返回错误码
* */
int listen_socket(int listenfd, int port); /**
* run_server - 运行服务器
* @listenfd:文件描述符
* */
void run_server(int listenfd); #ifdef __cplusplus
extern "C"
}
#endif
#endif
//serhelp.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/wait.h>
#include <signal.h>
#include "commsocket.h" /**
* sersocket_init - socket初始化
* @listenfd:文件描述符
* 成功返回0,失败返回错误码
* */
int sersocket_init(int *listenfd)
{
int ret = ;
if (listenfd == NULL)
{
ret = Sck_ParamErr;
printf("sersocket_init() params not correct !\n");
return ret;
}
//初始化socket环境
int fd = socket(AF_INET, SOCK_STREAM, );
if (fd == -)
{
ret = Sck_BaseErr;
perror("socket() err");
return ret;
}
*listenfd = fd;
return ret;
} /**
* listen_socket - 绑定端口号,监听套接字
* @listenfd:文件描述符
* @port:绑定的端口号
* 成功返回0,失败返回错误码
* */
int listen_socket(int listenfd, int port)
{
int ret = ;
if (listenfd < || port < || port > )
{
ret = Sck_ParamErr;
printf("listen_socket() params not correct !\n");
return ret;
}
//reuse addr
int optval = ;
ret = setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &optval,
sizeof(optval));
if (ret == -)
{
ret = Sck_BaseErr;
perror("setsockopt() err");
return ret;
}
//bind
struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(port);
addr.sin_addr.s_addr = inet_addr("127.0.0.1");
ret = bind(listenfd, (struct sockaddr *) &addr, sizeof(addr));
if (ret == -)
{
ret = Sck_BaseErr;
perror("bind() err");
return ret;
}
//listen
ret = listen(listenfd, SOMAXCONN);
if (ret == -)
{
ret = Sck_BaseErr;
perror("listen() err");
return ret;
}
return ret;
} /**
* handler - 信号捕捉函数
* @sign:信号码
* */
void handler(int sign)
{
if (sign == SIGPIPE)
{
printf("server accept SIGPIPE !\n");
}
} /**
* run_server - 运行服务器
* @listenfd:文件描述符
* */
void run_server(int listenfd)
{
int ret = ;
//屏蔽SIGPIPIE信号
if (signal(SIGPIPE, handler) == SIG_ERR)
{
printf("signal() failed !\n");
return;
}
//定义文件描述符集合
fd_set allsets;
FD_ZERO(&allsets);
fd_set readfds;
FD_ZERO(&readfds);
//定义客户端套接字池
char socketPond[] = { };
int i = ;
for (i = ; i < ; i++)
{
socketPond[i] = -;
}
//定义池子最后一个元素的下标
int maxindex = ;
//定义文件描述符中值最大的fd
int maxfd = listenfd;
//将监听套接字加入到集合中
FD_SET(listenfd, &allsets);
struct sockaddr_in peeraddr;
socklen_t peerlen = sizeof(struct sockaddr_in);
//定义接收缓冲区
char buf[MAXBUFSIZE] = { };
int buflen = ;
int conn = ;
int nread = ;
while ()
{
readfds = allsets;
do
{
nread = select(maxfd + , &readfds, NULL, NULL, NULL);
} while (nread == - && errno == EINTR);//屏蔽信号(重点)
if (nread == -)
{
perror("select() err");
return;
} else if (nread == )
{
//超时
continue;
} else if (nread > )
{
//执行操作
//1.处理服务器监听套接字
if (FD_ISSET(listenfd, &readfds))
{
//客户端有连接完成三次握手
memset(&peeraddr, , sizeof(struct sockaddr_in));
peerlen = sizeof(struct sockaddr_in);
conn = accept(listenfd, (struct sockaddr *) &peeraddr,
&peerlen);
if (conn == -)
{
perror("accept() err");
break;
}
printf("accept from %s\n", inet_ntoa(peeraddr.sin_addr));
//将客户端套接字加入池子中
for (i = ; i < ; i++)
{
if (socketPond[i] == -)
{
socketPond[i] = conn;
//数组最大下标后移
if (maxindex <= i)
maxindex = i + ;
break;
}
}
//如果超过最大连接数,直接关闭连接(重点)
if (i == )
{
close(conn);
continue;
}
//将客户端套接字加入到监听集合中
FD_SET(conn, &allsets);
//每新加一个连接,就更新最大套接字(重点)
if(conn>maxfd)
maxfd=conn;
//需要处理的事件数-1(重点)
if (--nread <= )
continue;
}
//2.客户端读事件
for (i = ; i < maxindex; i++)
{
if (socketPond[i] == -)
continue;
if (FD_ISSET(socketPond[i], &readfds))
{
//接收客户端信息
memset(buf, , sizeof(buf));
buflen = MAXBUFSIZE;
ret = socket_recv(socketPond[i], buf, &buflen);
if (ret == -)
{
//接收信息出错,关闭套接字
close(socketPond[i]);
//将该套接字移除池子
socketPond[i] = -;
//将该套接字移除监听集合
FD_CLR(conn, &allsets);
} else
{
//打印信息
fputs(buf, stdout);
//向客户端发送数据
ret = socket_send(socketPond[i], buf, buflen);
if (ret == -)
{
//发送数据出错,关闭套接字
close(socketPond[i]);
//将该套接字移除池子
socketPond[i] = -;
//将该套接字移除监听集合
FD_CLR(conn, &allsets);
}
}
//处理的事件数-1
if (--nread <= )
break;
}
}
}
}
return;
}
//服务器
#include "serhelp.h"
#include <stdio.h>
#include "commsocket.h" int main()
{
int ret = ;
int sockfd = ;
//初始化socket
ret = sersocket_init(&sockfd);
if (ret != )
{
printf("error message:%s\n", strsockerr(ret));
return -;
}
//监听
ret = listen_socket(sockfd, );
if (ret != )
{
printf("error message:%s\n", strsockerr(ret));
return -;
}
//运行
run_server(sockfd);
return ;
}
.SUFFIXES:.c .o
CC=gcc
SRCS1=mserver.c\
serhelp.c\
commsocket.c\
sockhelp.c
OBJS1=$(SRCS1:.c=.o)
EXEC1=mser
SRCS2=mclient.c\
clthelp.c\
commsocket.c\
sockhelp.c
OBJS2=$(SRCS2:.c=.o)
EXEC2=mcl start:$(OBJS1) $(OBJS2)
$(CC) -o $(EXEC1) $(OBJS1)
$(CC) -o $(EXEC2) $(OBJS2)
@echo "--------OK-----------"
.c.o:
$(CC) -Wall -g -o $@ -c $<
clean:
rm -f $(OBJS1)
rm -f $(OBJS2)
rm -f $(EXEC1)
rm -f $(EXEC2)
Linux socket多进程服务器框架三的更多相关文章
- Linux socket多进程服务器框架一
重点:socket共用方法中错误码的定义以及错误码的解析 底层辅助代码 //serhelp.h #ifndef _vxser #define _vxser #ifdef __cplusplus ext ...
- Linux socket多进程服务器框架二
客户端未解决Bug:子进程或者父进程退出的时候,我无法做到两个进程都调用clt_socket_Destory()方式释放socket句柄, 但是进程退出后,相应的资源也会释放,有一定影响,但是不大,以 ...
- linux socket高性能服务器处理框架
这个博客很多东西 http://blog.csdn.net/luozhonghua2014/article/details/37041765 思考一种高性能的服务器处理框架 1.首先需要一个内存池 ...
- LINUX环境并发服务器的三种实现模型
服务器设计技术有很多,按使用的协议来分有TCP服务器和UDP服务器.按处理方式来分有循环服务器和并发服务器. 1 循环服务器与并发服务器模型 在网络程序里面,一般来说都是许多客户对应一个服务器,为了 ...
- workerman是一个高性能的PHP socket服务器框架
workerman-chatorkerman是一款纯PHP开发的开源高性能的PHP socket服务器框架.被广泛的用于手机app.手游服务端.网络游戏服务器.聊天室服务器.硬件通讯服务器.智能家居. ...
- PHP socket 服务器框架集
1.Swoole:重新定义PHP PHP语言的高性能网络通信框架,提供了PHP语言的异步多线程服务器,异步TCP/UDP网络客户端,异步MySQL,数据库连接池,AsyncTask,消息队列,毫秒定时 ...
- [C++]Linux之多进程运行代码框架
声明:如需引用或者摘抄本博文源码或者其文章的,请在显著处注明,来源于本博文/作者,以示尊重劳动成果,助力开源精神.也欢迎大家一起探讨,交流,以共同进步- 0.0 多进程代码框架示例 /* @url: ...
- Linux socket 类封装 (面向对象方法)
/* * socketfactory.h * * Created on: 2014-7-19 * Author: root */ #ifndef SOCKETFACTORY_H_ #define SO ...
- 【LINUX/UNIX网络编程】之使用消息队列,信号量和命名管道实现的多进程服务器(多人群聊系统)
RT,使用消息队列,信号量和命名管道实现的多人群聊系统. 本学期Linux.unix网络编程的第三个作业. 先上实验要求: 实验三 多进程服务器 [实验目的] 1.熟练掌握进程的创建与终止方法: 2 ...
随机推荐
- Android无线开发的几种常用技术(阿里巴巴资深工程师原创分享)
完整的开发一个android移动App需要经过从分解需求.架构设计到开发调试.测试.上线发布等多个阶段,在发布后还会有产品功能上的迭代演进,此外还会面对性能.安全.无线网络质量等多方面的问题. 移动A ...
- Nutz中那些好用的工具类
Nutz 是国产的精品开源框架,它全无依赖,只以1兆多的身材,就可以实现SSH全部功能的90%以上.内容主要涵盖了:Ioc.Aop.MVC.Dao.Json等WEB开发的方方面面. 它不仅轻巧,而且 ...
- 转:JQuery.Ajax之错误调试帮助信息
今天发现一篇讲Ajax比较好的文章,汇总下,作为自己的知识储备. 下面是Jquery中AJAX参数详细列表: 参数名 类型 描述 url String (默认: 当前页地址) 发送请求的地址. typ ...
- 面试题整理:C#(一)
该系列持续更新,从网上以及身边收集的问题 1.可访问性级别有哪几种 public 访问不受限制.protected 访问仅限于包含类或从包含类派生的类型.internal 访问仅限于当前程序集.pro ...
- mysql metadata lock(二)
上一篇<mysql metadata lock(一)>介绍了为什么引入MDL,MDL作用以及MDL锁导致阻塞的几种典型场景,文章的最后还留下了一个小小的疑问.本文将更详细的介绍MDL,主要 ...
- Java中Date的比较(befor与after方法的缺陷)
java.util.Date中的before和after方法只会比较到Day,不管你的date是yyyy-MM-dd HH:mm:ss还是yyyy-MM-dd格式的.最好用getTime()来比较具体 ...
- spring 依赖注入(IOC DI)
依赖注入(IOC DI) 依赖注入的两种方式: 1. set注入 Spring要求使用set注入方式的时候,Bean需要提供一个无参数的构造方法.并提供一个属性的setter方法.例如: packag ...
- C语言(函数)学习之index、rindex
函数定义:char *index(const char *s, int c); 头文件: #include strings.h 函数说明:index()用来找出参数s 字符串中第一个出现的参数c ...
- 010 使用netmap API接管网卡,接收数据包,回应ARP请求
一.本文目的: 上一节中,我们已经在CentOS 6.7 上安装好了netmap,也能接收和发送包了,这节我们来调用netmap中的API,接管网卡,对网卡上收到的数据包做分析,并回应ARP请求. 二 ...
- 新手必学的java报表开发工具FineReport实用技巧
1.在制作模板时,如何将报表中的值传递到超链接网页呢? 在项目中以frame方法把F1.CPT放到项目的页面中,对F1.CPT做网络报表超链接F2.CPT,然后在F2.cpt页面中,做个超链接的网页, ...