免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 8291 | 回复: 10
打印 上一主题 下一主题

epoll的ET和LT问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2009-10-13 11:48 |只看该作者 |倒序浏览
在网上找到一段代码来试验epoll的ET模式和LT模式的区别,服务端代码如下:
#include <iostream>
#include <sys/socket.h>
#include <sys/epoll.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include<sys/types.h>

using namespace std;

#define MAXLINE 5
#define OPEN_MAX 100
#define LISTENQ 20
#define SERV_PORT 5000
#define INFTIM 1000

void setnonblocking(int sock)
{
    int opts;
    opts=fcntl(sock,F_GETFL);
    if(opts<0)
    {
        perror("fcntl(sock,GETFL)");
        exit(1);
    }
    opts = opts|O_NONBLOCK;
    if(fcntl(sock,F_SETFL,opts)<0)
    {
        perror("fcntl(sock,SETFL,opts)");
        exit(1);
    }   
}

int main()
{
    int i, maxi, listenfd, connfd, sockfd,epfd,nfds;
    ssize_t n;
    char line[MAXLINE];
    socklen_t clilen;
    //声明epoll_event结构体的变量,ev用于注册事件,数组用于回传要处理的事件
    struct epoll_event ev,events[20];
    //生成用于处理accept的epoll专用的文件描述符
    epfd=epoll_create(256);
    struct sockaddr_in clientaddr;
    struct sockaddr_in serveraddr;
    clilen=sizeof(clientaddr);
    listenfd = socket(AF_INET, SOCK_STREAM, 0);
    //把socket设置为非阻塞方式
   setnonblocking(listenfd);
    //设置与要处理的事件相关的文件描述符
    ev.data.fd=listenfd;
    //设置要处理的事件类型
    ev.events=EPOLLIN|EPOLLET;
    //ev.events=EPOLLIN;
    //注册epoll事件
    epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,&ev);
    bzero(&serveraddr, sizeof(serveraddr));
    serveraddr.sin_family = AF_INET;
    char *local_addr="127.0.0.1";
    inet_aton(local_addr,&(serveraddr.sin_addr));//htons(SERV_PORT);
    serveraddr.sin_port=htons(SERV_PORT);
    bind(listenfd,(struct sockaddr *)&serveraddr, sizeof(serveraddr));
    listen(listenfd, LISTENQ);
    maxi = 0;
    for ( ; ; ) {
        //等待epoll事件的发生
        nfds=epoll_wait(epfd,events,20,500);
        //处理所发生的所有事件     
        for(i=0;i<nfds;++i)
        {
            if(events[i].data.fd==listenfd)
            {
                connfd = accept(listenfd,(struct sockaddr *)&clientaddr, &clilen);
                if(connfd<0){
                    perror("connfd<0");
                    exit(1);
                }
                setnonblocking(connfd);
                char *str = inet_ntoa(clientaddr.sin_addr);
                cout << "accapt a connection from " << str << endl;
                //设置用于读操作的文件描述符
                ev.data.fd=connfd;
                //设置用于注测的读操作事件
                ev.events=EPOLLIN|EPOLLET;
                //ev.events=EPOLLIN;
                //注册ev
                epoll_ctl(epfd,EPOLL_CTL_ADD,connfd,&ev);
            }
            else if(events[i].events&EPOLLIN)
            {
                cout << "EPOLLIN" << endl;
                if ( (sockfd = events[i].data.fd) < 0)
                    continue;
                if ( (n = read(sockfd, line, MAXLINE)) < 0) {
                    if (errno == ECONNRESET) {
                        close(sockfd);
                        events[i].data.fd = -1;
                    } else
                        std::cout<<"readline error"<<std::endl;
                } else if (n == 0) {
                    close(sockfd);
                    events[i].data.fd = -1;
                }
                line[n] = '\0';
                cout << "read " << line << endl;
                //设置用于写操作的文件描述符
                ev.data.fd=sockfd;
                //设置用于注测的写操作事件
                ev.events=EPOLLOUT|EPOLLET;
                //修改sockfd上要处理的事件为EPOLLOUT
                epoll_ctl(epfd,EPOLL_CTL_MOD,sockfd,&ev);
            }
            else if(events[i].events&EPOLLOUT)
            {   
                sockfd = events[i].data.fd;
                write(sockfd, line, n);
                //设置用于读操作的文件描述符
                ev.data.fd=sockfd;
                //设置用于注测的读操作事件
                ev.events=EPOLLIN|EPOLLET;
                //修改sockfd上要处理的事件为EPOLIN
                epoll_ctl(epfd,EPOLL_CTL_MOD,sockfd,&ev);
            }
        }
    }
    return 0;
}
客户端的代码用perl写的如下:
#!/usr/bin/perl

use IO::Socket;

my $host = "127.0.0.1";
my $port = 5000;

my $socket = IO::Socket::INET->new("$host:$port") or die "create socket error $@";
my $msg_out = "1234567890";
print $socket $msg_out;
print "now send over, go to sleep\n";

while (1)
{
    sleep(1);
}


“运行server和client发现,server仅仅读取了5字节的数据,而client其实发送了10字节的数据,也就是说,server仅当第一次监听到了EPOLLIN事件,由于没有读取完数据,而且采用的是ET模式,状态在此之后不发生变化,因此server再也接收不到EPOLLIN事件了.”
上面这段话是作者的,可我在用这段代码做试验的时候,发现服务端会触发两次,也就是分别读取了12345,然后再读取67890,然后我在强制关闭客户端的话,还会在输出一次67890,然后我改成LT模式得到的结果也是一样的,在使用ET模式的时候为什么还会读取67890呢?各位大虾帮帮忙,小弟在此谢过了!!!man 手册也看了,也上网搜集一下ET和LT的区别,按照我自己理解这段代码也应该只能得到前面的12345才对的。下面是我执行的结果,各位帮帮忙,谢谢了

[[i] 本帖最后由 lxjeanse 于 2009-10-14 00:01 编辑 [/i]]

剪辑.jpg (35.57 KB, 下载次数: 27)

剪辑.jpg

论坛徽章:
0
2 [报告]
发表于 2009-10-13 13:39 |只看该作者
manual上那个例子已经说的很明白了,你详细看下应该可以的

论坛徽章:
0
3 [报告]
发表于 2009-10-13 13:44 |只看该作者
原帖由 @sky 于 2009-10-13 13:39 发表
manual上那个例子已经说的很明白了,你详细看下应该可以的

看的话都会明白啊,但是具体试验的时候就不行的了!

论坛徽章:
0
4 [报告]
发表于 2009-10-13 17:35 |只看该作者

回复 #3 lxjeanse 的帖子

客户端close(socket),会出发EPOLLIN, 强制关闭客户端程序的时候,我这里会出发EPOLLERR, 至于你那个为什么会出发两次EPOLLIN,我也期待高手解答!你用程序写个客户端试试!

论坛徽章:
0
5 [报告]
发表于 2009-10-13 20:00 |只看该作者
原帖由 齐得龙强更强 于 2009-10-13 17:35 发表
客户端close(socket),会出发EPOLLIN, 强制关闭客户端程序的时候,我这里会出发EPOLLERR, 至于你那个为什么会出发两次EPOLLIN,我也期待高手解答!你用程序写个客户端试试!

恩,谢谢了,客户端关闭socket是会触发EPOLLIN的,我就是不明白怎么会触发会在没有关闭客户端的时候触发了输出12345之后还会触发67890,一会写个客户端看看!

论坛徽章:
0
6 [报告]
发表于 2009-10-13 21:04 |只看该作者
LZ,你用ET写的server在关闭客户端的时候才收到的后面几个数据吧?那是因为客户端关闭的时候,server也会收到io消息,这时就会去接收剩下的数据了.

论坛徽章:
0
7 [报告]
发表于 2009-10-13 22:12 |只看该作者
原帖由 converse 于 2009-10-13 21:04 发表
LZ,你用ET写的server在关闭客户端的时候才收到的后面几个数据吧?那是因为客户端关闭的时候,server也会收到io消息,这时就会去接收剩下的数据了.

恩,不是的,会收到两次67890的,就是我在没关闭客户端的时候也会触发,所以我很纳闷

论坛徽章:
324
射手座
日期:2013-08-23 12:04:38射手座
日期:2013-08-23 16:18:12未羊
日期:2013-08-30 14:33:15水瓶座
日期:2013-09-02 16:44:31摩羯座
日期:2013-09-25 09:33:52双子座
日期:2013-09-26 12:21:10金牛座
日期:2013-10-14 09:08:49申猴
日期:2013-10-16 13:09:43子鼠
日期:2013-10-17 23:23:19射手座
日期:2013-10-18 13:00:27金牛座
日期:2013-10-18 15:47:57午马
日期:2013-10-18 21:43:38
8 [报告]
发表于 2009-10-14 09:50 |只看该作者
好像是EPOLLOUT触发时,若有数据可读,会同时附带触发EPOLLIN。
两次输出67890应该是
if ( (n = read(sockfd, line, MAXLINE)) < 0) {
这个分支继续执行的缘故, line[n] = '\0';实际越界访问了

论坛徽章:
0
9 [报告]
发表于 2009-10-14 10:28 |只看该作者
出发两次是由你的EPOLL_CTL_MOD造成的,

第一次IN , 你read了5bytes, 然后你改成OUT
触发OUT, 你write,然后改成IN, 这个时候由于tcp缓冲区还有几个bytes,当然又会触发IN

论坛徽章:
0
10 [报告]
发表于 2009-10-14 10:30 |只看该作者
还有就是楼主的代码写得有些问题,你应该完全读懂epoll的man,然后按照建议来写你的代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP