lishou564 2016-11-14 03:19 采纳率: 0%
浏览 2272

linux下socket编程 select只监测读事件, 为什么还会出现读阻塞

我用select监测读时间,但是read出现了阻塞

maxfd = iListenFdIP;
maxi = -1;
for(i=0 ; i<CONNCT_NUM_MAX; ++i)
{
    client[i][0] = -1;
}//for
FD_ZERO(&allset);
FD_SET(iListenFdIP , &allset);
i = 0;
while(1){
    rset = allset;
    nready = select(maxfd+1 , &rset , NULL , NULL , NULL);

// DEBUG(LOG_DEBUG,"nread:%d\n",nready);
if(FD_ISSET(iListenFdIP , &rset))
{
/*接收客户端的请求*/
clilen = sizeof(cliaddr);

// DEBUG(LOG_DEBUG,"\naccpet connection~\n");

        if((connfd = accept( iListenFdIP, (struct sockaddr *)&cliaddr , &clilen)) < 0)
        {
            DEBUG(LOG_ERROR,"accept error.\n");
            continue;
        }//if       

        DEBUG(LOG_DEBUG,"accpet a new client: %s:%d\n", inet_ntoa(cliaddr.sin_addr) , cliaddr.sin_port);

        /*将客户链接套接字描述符添加到数组*/
        for(i=0 ; i<CONNCT_NUM_MAX ; ++i)
        {
            if(client[i][0] < 0)
            {
                client[i][0] = connfd;
                client[i][1] = cliaddr.sin_port;
                break;
            }//if
        }//for

        if(CONNCT_NUM_MAX == i)
        {
            DEBUG(LOG_ERROR,"too many connection.\n");
        }//if
        else
        {
            FD_SET(connfd , &allset);
            if(connfd > maxfd)
                maxfd = connfd;
            if(i > maxi)
                maxi = i;
        }


        if(--nready < 0)
            continue;
    }//if

// DEBUG(LOG_DEBUG,"maxi:%d\n",maxi);
for(i=0; i<=maxi ; ++i)
{
memset(caBuff,0x0,sizeof(caBuff));
memset(sndBuff,0x0,sizeof(sndBuff));

memset(caSplitBuff,0x0,sizeof(caSplitBuff));

if((sockfd = client[i][0]) < 0)
continue;
if(FD_ISSET(sockfd , &rset))
{
/*处理客户请求*/
//DEBUG(LOG_DEBUG,"\nreading the socket~~~ \n");

            if((n = read(sockfd , caBuff , sizeof(caBuff))) <= 0)
            {
                DEBUG(LOG_DEBUG,"client[%d] is close\n",i);
                close(sockfd);
                FD_CLR(sockfd , &allset);
                client[i][0] = -1;
            }//if
            else{

                char* cpPosNow = caBuff;

                if ( NULL == (cpPos = strchr(cpPosNow, '\n'))){
                    DEBUG(LOG_ERROR,"data:%s  can't find '\n' \n",cpPosNow);
                    continue;
                }
                iPkgLen = cpPos - cpPosNow;
                memcpy(caSplitBuff, cpPosNow, iPkgLen);

                inet_aton(caSplitBuff,&addr.sin_addr);

                memset(caSplitBuff,0x0,sizeof(caSplitBuff));
                memcpy(&tmpIP,&addr.sin_addr,4);
                //tmpIP = htonl(tmpIP);
                memcpy(caSplitBuff,&tmpIP,4);
                if(NULL == (pIpMapHashNode = ipmap_hash_table_lookup((const unsigned char *)caSplitBuff))){
                    res = send(sockfd, "IPOFFLINE\n",strlen("IPOFFLINE\n") , 0);
                    if(res < 0){
                        close(sockfd);
                        FD_CLR(sockfd , &allset);
                        client[i][0] = -1;
                        DEBUG(LOG_DEBUG,"Send ERROR:Client is close,ErrorNum = %d, ErrorMsg = %s\n", errno, strerror(errno));
                    }
                }
                else{

                    DEBUG(LOG_DEBUG,"recv IPV4:%d.%d.%d.%d username:%s dstprot:%d\n",
                                        caSplitBuff[0],caSplitBuff[1],caSplitBuff[2],caSplitBuff[3],pIpMapHashNode->UserName,client[i][1]);
                    memcpy(sndBuff, pIpMapHashNode->UserName, strlen(pIpMapHashNode->UserName));
                    strcat(sndBuff,"\n");
                    res = send(sockfd, sndBuff, strlen(pIpMapHashNode->UserName)+1, 0);
                    if(res <=  0){
                        close(sockfd);
                        FD_CLR(sockfd , &allset);
                        client[i][0] = -1;
                        DEBUG(LOG_DEBUG,"ERROR:Send Client is close,ErrorNum = %d, ErrorMsg = %s\n", errno, strerror(errno));                           
                    }
                }
            }
            if(--nready <= 0)
                break;
        }
    }
 }
 close(iListenFdIP);
 return;     
  • 写回答

1条回答 默认 最新

  • oyljerry 2016-11-14 12:58
    关注

    要设置你的socket为非阻塞模式

    评论

报告相同问题?

悬赏问题

  • ¥15 名为“Product”的列已属于此 DataTable
  • ¥15 安卓adb backup备份应用数据失败
  • ¥15 eclipse运行项目时遇到的问题
  • ¥15 关于#c##的问题:最近需要用CAT工具Trados进行一些开发
  • ¥15 南大pa1 小游戏没有界面,并且报了如下错误,尝试过换显卡驱动,但是好像不行
  • ¥15 没有证书,nginx怎么反向代理到只能接受https的公网网站
  • ¥50 成都蓉城足球俱乐部小程序抢票
  • ¥15 yolov7训练自己的数据集
  • ¥15 esp8266与51单片机连接问题(标签-单片机|关键词-串口)(相关搜索:51单片机|单片机|测试代码)
  • ¥15 电力市场出清matlab yalmip kkt 双层优化问题