python epoll 并发_高并发的epoll+线程池,epoll在线程池内

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

//stl head

#include //包含hash_map 的头文件

//#include //stl的map

using namespace std; //std 命名空间

using namespace __gnu_cxx; //而hash_map是在__gnu_cxx的命名空间里的

int init_thread_pool(int threadNum);

void *epoll_loop(void* para);

void *check_connect_timeout(void* para);

struct sockStruct

{

time_t time;

unsigned int* recvBuf;

};

//hash-map

//hash_mapsock_map;

hash_map sock_map;

#define MAXRECVBUF 4096

#define MAXBUF MAXRECVBUF+10

int fd_Setnonblocking(int fd)

{

int op;

op=fcntl(fd,F_GETFL,0);

fcntl(fd,F_SETFL,op|O_NONBLOCK);

return op;

}

void on_sigint(int signal)

{

exit(0);

}

/*

handle_message - 处理每个 socket 上的消息收发

*/

int handle_message(int new_fd)

{

char buf[MAXBUF + 1];

char sendbuf[MAXBUF+1];

int len;

/* 开始处理每个新连接上的数据收发 */

bzero(buf, MAXBUF + 1);

/* 接收客户端的消息 */

//len = recv(new_fd, buf, MAXBUF, 0);

int nRecvBuf = MAXRECVBUF; //设置为32K

setsockopt(new_fd, SOL_SOCKET, SO_RCVBUF, ( const char* )&nRecvBuf, sizeof(int));

len=recv(new_fd,&buf, MAXBUF,0);

//--------------------------------------------------------------------------------------------

//这块为了使用ab测试

char bufSend[1000] = {0};

sprintf(bufSend,"HTTP/1.0 200 OK\r\nContent-type: text/plain\r\n\r\n%s","Hello world!\n");

send(new_fd,bufSend,strlen(buf),0);

//--------------------------------------------------------------------------------------------

if (len > 0){

//printf ("%d接收消息成功:'%s',共%d个字节的数据\n", new_fd, buf, len);

//hash-map

hash_map::iterator it_find;

it_find = sock_map.find(new_fd);

if(it_find == sock_map.end()){

//新的网络连接,申请新的接收缓冲区,并放入map中

//printf("new socket %d\n", new_fd);

sockStruct newSockStruct;

newSockStruct.time = time((time_t*)0);

newSockStruct.recvBuf = (unsigned int*)malloc(1000);

memset(newSockStruct.recvBuf, 0, 1000);

strcat((char*)newSockStruct.recvBuf, buf);

sock_map.insert(pair(new_fd, newSockStruct));

}else{

//网络连接已经存在,找到对应的数据缓冲区,将接收到的数据拼接到数据缓冲区中

//printf("socket %d exist!\n", it_find->first);

(it_find->second).time = time((time_t*)0); //时间更改

char* bufSockMap = (char*)(it_find->second).recvBuf; //数据存储

strcat(bufSockMap, buf);

//printf("bufSockMap:%s\n", bufSockMap);

}

}

else {

if (len < 0)

printf ("消息接收失败!错误代码是%d,错误信息是'%s'\n",

errno, strerror(errno));

else {

//将socket从map中移除

/*

hash_map::iterator it_find;

it_find = sock_map.find(new_fd);

sock_map.erase(it_find);

*/

printf("client %d quit!\n",new_fd);

}

//close(new_fd);

return -1;

}

/* 处理每个新连接上的数据收发结束 */

//关闭socket的时候,要释放接收缓冲区。

hash_map::iterator it_find;

it_find = sock_map.find(new_fd);

free((it_find->second).recvBuf);

sock_map.erase(it_find);

close(new_fd);

return len;

}

int listenfd;

int sock_op=1;

struct sockaddr_in address;

struct epoll_event event;

struct epoll_event events[1024];

int epfd;

int n;

int i;

char buf[512];

int off;

int result;

char *p;

int main(int argc,char* argv[])

{

init_thread_pool(1);

signal(SIGPIPE,SIG_IGN);

signal(SIGCHLD,SIG_IGN);

signal(SIGINT,&on_sigint);

listenfd=socket(AF_INET,SOCK_STREAM,0);

setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&sock_op,sizeof(sock_op));

memset(&address,0,sizeof(address));

address.sin_addr.s_addr=htonl(INADDR_ANY);

address.sin_port=htons(8006);

bind(listenfd,(struct sockaddr*)&address,sizeof(address));

listen(listenfd,1024);

fd_Setnonblocking(listenfd);

epfd=epoll_create(65535);

memset(&event,0,sizeof(event));

event.data.fd=listenfd;

event.events=EPOLLIN|EPOLLET;

epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,&event);

while(1){

sleep(1000);

}

return 0;

}

/*************************************************

* Function: * init_thread_pool

* Description: * 初始化线程

* Input: * threadNum:用于处理epoll的线程数

* Output: *

* Others: * 此函数为静态static函数,

*************************************************/

int init_thread_pool(int threadNum)

{

int i,ret;

pthread_t threadId;

//初始化epoll线程池

for ( i = 0; i < threadNum; i++)

{

ret = pthread_create(&threadId, 0, epoll_loop, (void *)0);

if (ret != 0)

{

printf("pthread create failed!\n");

return(-1);

}

}

ret = pthread_create(&threadId, 0, check_connect_timeout, (void *)0);

return(0);

}

/*************************************************

* Function: * epoll_loop

* Description: * epoll检测循环

* Input: *

* Output: *

* Others: *

*************************************************/

static int count111 = 0;

static time_t oldtime = 0, nowtime = 0;

void *epoll_loop(void* para)

{

while(1)

{

n=epoll_wait(epfd,events,4096,-1);

//printf("n = %d\n", n);

if(n>0)

{

for(i=0;i

{

if(events[i].data.fd==listenfd)

{

while(1)

{

event.data.fd=accept(listenfd,NULL,NULL);

if(event.data.fd>0)

{

fd_Setnonblocking(event.data.fd);

event.events=EPOLLIN|EPOLLET;

epoll_ctl(epfd,EPOLL_CTL_ADD,event.data.fd,&event);

}

else

{

if(errno==EAGAIN)

break;

}

}

}

else

{

if(events[i].events&EPOLLIN)

{

//handle_message(events[i].data.fd);

char recvBuf[1024] = {0};

int ret = 999;

int rs = 1;

while(rs)

{

ret = recv(events[n].data.fd,recvBuf,1024,0);// 接受客户端消息

if(ret < 0)

{

//由于是非阻塞的模式,所以当errno为EAGAIN时,表示当前缓冲区已无数据可//读在这里就当作是该次事件已处理过。

if(errno == EAGAIN)

{

printf("EAGAIN\n");

break;

}

else{

printf("recv error!\n");

epoll_ctl(epfd, EPOLL_CTL_DEL, events[i].data.fd, &event);

close(events[i].data.fd);

break;

}

}

else if(ret == 0)

{

// 这里表示对端的socket已正常关闭.

rs = 0;

}

if(ret == sizeof(recvBuf))

rs = 1; // 需要再次读取

else

rs = 0;

}

if(ret>0){

count111 ++;

struct tm *today;

time_t ltime;

time( &nowtime );

if(nowtime != oldtime){

printf("%d\n", count111);

oldtime = nowtime;

count111 = 0;

}

char buf[1000] = {0};

sprintf(buf,"HTTP/1.0 200 OK\r\nContent-type: text/plain\r\n\r\n%s","Hello world!\n");

send(events[i].data.fd,buf,strlen(buf),0);

// CGelsServer Gelsserver;

// Gelsserver.handle_message(events[i].data.fd);

}

epoll_ctl(epfd, EPOLL_CTL_DEL, events[i].data.fd, &event);

close(events[i].data.fd);

}

else if(events[i].events&EPOLLOUT)

{

sprintf(buf,"HTTP/1.0 200 OK\r\nContent-type: text/plain\r\n\r\n%s","Hello world!\n");

send(events[i].data.fd,buf,strlen(buf),0);

/*

if(p!=NULL)

{

free(p);

p=NULL;

}

*/

close(events[i].data.fd);

}

else

{

close(events[i].data.fd);

}

}

}

}

}

}

/*************************************************

* Function: * check_connect_timeout

* Description: * 检测长时间没反应的网络连接,并关闭删除

* Input: *

* Output: *

* Others: *

*************************************************/

void *check_connect_timeout(void* para)

{

hash_map::iterator it_find;

for(it_find = sock_map.begin(); it_find!=sock_map.end(); ++it_find){

if( time((time_t*)0) - (it_find->second).time > 120){ //时间更改

free((it_find->second).recvBuf);

sock_map.erase(it_find);

close(it_find->first);

}

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在socket epoll高并发项目,我们可以使用以下方法来测试高并发性: 1. 压力测试:可以使用工具如Apache JMeter或wrk等进行压力测试,模拟多个并发请求发送到服务器。可以设置并发连接数和请求频率,观察服务器的响应时间和处理能力。 2. 性能测试:可以使用工具如Apache Bench或siege等进行性能测试,发送多个并发请求并记录服务器的响应时间、吞吐量等指标。可以通过调整并发连接数和请求的大小来测试服务器的性能极限。 3. 负载均衡测试:如果项目使用了负载均衡器来分流请求,可以模拟多个并发请求发送到负载均衡器,观察负载均衡器的转发能力和服务器的响应时间。 4. 异常情况测试:可以模拟网络延迟、断连、异常数据等异常情况,观察服务器的容错能力和恢复能力。 5. 数据库测试:如果项目有涉及数据库的操作,可以模拟并发读写请求,观察数据库的性能和并发处理能力。 6. 监控和分析:在测试过程,可以使用监控工具来实时监测服务器的CPU、内存、网络等指标,以及检查是否有内存泄漏或资源泄漏等问题。 通过以上的测试方法和手段,我们可以评估高并发项目的性能和稳定性,找出性能瓶颈和优化空间,提高系统的并发处理能力。 ### 回答2: 在进行socket epoll高并发项目的测试时,可以采取以下几个步骤来测试高并发性: 1. 设计并发测试场景:根据项目的需求和设计,确定需要模拟的并发用户数、每个用户的请求频率和请求类型。可以使用工具如Apache JMeter或自行编写脚本来模拟并发请求。 2. 配置并发环境:在测试机器上进行并发测试,需要增加网络带宽、增加计算资源,比如使用高性能的服务器和网络设备,确保能够支持大量并发连接。 3. 编写测试程序:根据项目的需求,在测试程序实现模拟并发请求的逻辑,通过socket epoll模型建立大量并发连接,并发送模拟请求进行测试。 4. 监控并发连接数和响应时间:使用系统工具如netstat、top等来监控服务器端的并发连接数和系统资源使用情况。同时,使用性能监控工具如zabbix、grafana等来监控服务器的吞吐量、响应时间等指标。 5. 数据验证和压力测试:在并发测试,确保数据的一致性,对接收到的响应进行验证。并逐步增加并发连接数,直至达到系统的极限,观察系统响应时间的变化情况和可能出现的性能瓶颈。 6. 多样化的测试场景:在测试过程,可以尝试不同的测试场景,如不同的请求类型、不同大小的数据包等,验证系统在各种情况下的高并发性能。 7. 异常处理:在测试,需要注意处理一些异常情况,如客户端异常断开连接、网络异常等,确保系统对异常情况的处理能力。 通过以上步骤,可以对socket epoll高并发项目进行有效的测试,找出系统的性能瓶颈,及时进行优化和调整,提升系统的高并发性能。 ### 回答3: 在socket epoll高并发项目,为了测试高并发性能,可以采取以下几种方式: 1. 压力测试工具:使用一些专业的压力测试工具,如JMeter、Apache Bench或wrk等,来模拟大量的并发请求。可以设置并发数、每秒请求数和总请求量等参数,对系统进行压力测试,观察系统在高并发情况下的性能表现。 2. 自动化测试脚本:编写自动化测试脚本,通过多线程或多进程进行模拟并发请求,向服务器发送大量的请求。可以使用Python的模块,如requests、multiprocessing等,实现并发请求的测试。 3. 网络负载生成工具:使用网络负载生成工具,比如Locust、Gatling等,来模拟真实的网络负载情况。可以设置请求频率、并发数和持续时间等参数,模拟多种场景下的高并发情况。 4. 并发性能监控工具:使用一些并发性能监控工具,如Grafana、Prometheus等,来监控系统的并发性能。通过收集CPU、内存、网络等指标数据,可以分析和评估系统在高并发场景下的性能瓶颈。 5. 随机性测试:在测试过程,引入随机性因素,模拟真实场景下的随机请求。可以设计不同类型的请求和不同请求参数的组合,观察系统在随机请求下的并发性能表现。 需要注意的是,在进行高并发性能测试时,要合理设置测试参数,以真实场景为依据,同时监控系统各项指标,及时发现并解决性能瓶颈。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值