Chinaunix首页 | 论坛 | 博客
  • 博客访问: 912537
  • 博文数量: 299
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 2493
  • 用 户 组: 普通用户
  • 注册时间: 2014-03-21 10:07
个人简介

Linux后台服务器编程。

文章分类

全部博文(299)

文章存档

2015年(2)

2014年(297)

分类: 网络与安全

2014-08-04 08:43:27

最近项目组中有个同事使用epoll+多线程实现了一个简单的服务器,但是经过压测后,发现如果使用边缘触发模式的话,就会出现丢包现象,水平触发设置等待时间确实解决了丢包问题,但却影响了服务器的性能。所以我就写了一个epoll+多进程的模型,代码如下:
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include  
#include
#include
#include  
using namespace std; 
#define MAX_EVENTS 10000 
 
 
void Process(int listenFd);
 
struct shmstruct
{
    int count;
};
 
int main(int argc, char **argv) 
    short port = 6666; // default port 
    if(argc == 2){ 
        port = atoi(argv[1]); 
    }
 
    shm_unlink("test");
    int fd1 = shm_open("test",O_RDWR|O_CREAT|O_EXCL,666);
    struct shmstruct* ptr;
    ftruncate(fd1,sizeof(struct shmstruct));
    ptr = (struct shmstruct*)mmap(NULL,sizeof(struct shmstruct),PROT_READ|PROT_WRITE,MAP_SHARED,fd1,0);
    close(fd1);
 
    int listenFd = socket(AF_INET, SOCK_STREAM, 0);
    fcntl(listenFd, F_SETFL, O_NONBLOCK); // 设置非阻塞方式
    sockaddr_in sin; 
    bzero(&sin, sizeof(sin)); 
    sin.sin_family = AF_INET; 
    sin.sin_addr.s_addr = INADDR_ANY; 
    sin.sin_port = htons(port); 
    bind(listenFd, (const sockaddr*)&sin, sizeof(sin)); 
    listen(listenFd, 5); 
 
    for (int i=0; i<5 ;i++)
    {
        pid_t pid = fork();
        if (pid == 0)   // 子进程
        {
            Process(listenFd); // 工作进程
            return 0;
        }
    }
    while (true)
    {
        sleep(10);
    }
 
    return 0; 
}  
 
 
void Process(int listenFd)
{
    int fd1 = shm_open("test",O_RDWR,666);
    struct shmstruct* ptr;
    ptr = (struct shmstruct*)mmap(NULL,sizeof(struct shmstruct),PROT_READ|PROT_WRITE,MAP_SHARED,fd1,0);
    close(fd1);
    struct epoll_event ev, events[MAX_EVENTS];
    //生成用于处理accept的 epoll专用的文件描述符
    int epfd = epoll_create( MAX_EVENTS );
    //设置与要处理的事件相关的文件描述符
    ev.data.fd = listenFd;
    //设置要处理的事件类型
    ev.events = EPOLLIN | EPOLLET;
    //注册epoll事件
    if ( epoll_ctl( epfd, EPOLL_CTL_ADD, listenFd, &ev ) < 0 )
    {
        printf( "worker epoll_ctl error = %s.", strerror(errno) );
        exit(1);
    }
 
    while (true)
    {
        // 等待epoll事件的发生
        int nfds = epoll_wait( epfd, events, MAX_EVENTS, -1 );
        // 处理所发生的所有事件
        for( int i=0; i
        {
            if( events[i].data.fd == listenFd )
            {
                socklen_t clilen;
                struct sockaddr_in clientaddr;
 
                int sockfd = accept( listenFd, (sockaddr *)&clientaddr, &clilen );
                if( sockfd < 0 )
                {
                    continue;
                }
 
                // 设置非阻塞
                if (fcntl( sockfd, F_SETFL, fcntl( sockfd, F_GETFD, 0)|O_NONBLOCK) == -1)
                {
                    continue;
                }
                //设置用于读操作的文件描述符
                ev.data.fd = sockfd;
                //设置用于注测的读操作事件
                ev.events = EPOLLIN | EPOLLET;
                //注册ev
                epoll_ctl( epfd, EPOLL_CTL_ADD, sockfd, &ev);
            }
            else if (events[i].events & EPOLLIN)
            {
                int sockfd = events[i].data.fd;
                if ( sockfd < 0 )
                {
                    continue;
                }
 
                char buf[1024] = {0};
                // 开始处理每个新连接上的数据收发
                bzero( buf, sizeof(buf) );
                int len = read( sockfd, buf, 1023 );
                if ( len < 0)
                {
                    if (errno == ECONNRESET)
                    {
                        close(sockfd);
                        events[i].data.fd = -1;
                    }
                    else
                    {
                        printf( "worker read data error = %s.", strerror(errno) );
                    }
                }
                else if ( len == 0 )
                {
                    events[i].data.fd = -1;
                }
                else
                {
                    if( send(sockfd, "asdf", 4, 0) < 0)   
                    {
                        printf("send msg error: %s(errno: %d)\n", strerror(errno), errno);
                        exit(0);
                    }
                    printf("count:%d\n",++ptr->count);
                }
            }
        }
    }
}


注:代码在redhat下能正常编译,如果其他平台,可能需要修改共享内存部分的代码,因为有些linux平台共享内存的创建路径需要写绝对路径。

  通过测试,压测的结果与子进程数成线性关系,也与服务器的配置有一定关系,我用的机器在5个子进程下,能达到5000的并发不丢包。其实如果在for循环内使用线程池进行业务处理的话,可能启用的子进程数不需要很多就能达到很大数量的并发,这个epoll+多进程+线程池也是目前多数服务器使用的epoll模型。等有时间了,再写个加入线程池的模型也测试测试。

阅读(1824) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~