socket阻塞与非阻塞实验(recv或read读返回值)(带实例)

实验分析了在服务器端,listenfd设置为阻塞与非阻塞模式对accept和recv操作的影响。当listenfd非阻塞时,服务器在没有客户端连接时会频繁检查,导致CPU占用率高。而设置为阻塞模式时,服务器会等待客户端连接,降低CPU使用。在recv操作中,非阻塞模式的cfd在无数据时会立即返回错误,而阻塞模式会等待数据到达。实验表明,服务器端的listenfd通常应设置为阻塞模式以优化资源利用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

参考该实验报告

网络软件设计课程实验(电子科技大学)多路复用-阻塞与非阻塞实验指导书.doc 

 

1、客户端没有与服务器端连接,测试accept的阻塞情况

1.1服务器端非阻塞 listenfd

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<unistd.h>
#include<pthread.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h>
void* ser_fun(void* arg)
{
    pthread_detach(pthread_self());
    long cfd = (long)arg;
    int nread;
    char buf[1024];
    while (1) {
        printf("zhunbei read..\n");
        nread = read(cfd,buf,1024);
        if(nread <=0)
        {
            printf("read:%s,errno=%d",strerror(errno),errno);
            close(cfd);
            pthread_exit(NULL);
        }
        printf("buf=%s\n",buf);
        sleep(1);
    }
}
int main()
{
    int sfd= -1,cfd = -1;
    pthread_t cid;
    struct sockaddr_in s_sockaddr,c_sockaddr;
    int c_sockaddr_len = sizeof(c_sockaddr);//必须赋值
    s_sockaddr.sin_family = AF_INET;
    s_sockaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    s_sockaddr.sin_port = htons(8888);
    sfd = socket(AF_INET,SOCK_STREAM,0);
    if(sfd<0)
    {
        perror("socket:");
        exit(-1);
    }
    //
    int flags = fcntl(sfd, F_GETFL, 0);                       //获取文件的flags值。
    fcntl(sfd, F_SETFL, flags | O_NONBLOCK);   //设置成非阻塞模式;

    if((bind(sfd,(struct sockaddr*)&s_sockaddr,sizeof(s_sockaddr))<0))//
    {
        perror("bind:");
        close(sfd);
        exit(-1);
    }
    if(listen(sfd,2)<0)
    {
        perror("listen:");
        close(sfd);
        exit(-1);
    }
    while(1)
    {
        if((cfd = accept(sfd,(struct sockaddr*)&c_sockaddr,(socklen_t*)&c_sockaddr_len))<0)
        {
            perror("accept");
            usleep(10);
            continue;
        }
        printf("client link,ip=%s,port= %d,cfd =%d\n",inet_ntoa(c_sockaddr.sin_addr),
               ntohs(c_sockaddr.sin_port),cfd);
        pthread_create(&cid,NULL,ser_fun,(void*)cfd);
    }
    return 0;
}


状态处于睡眠和运行中交替执行,且cpu占用率高。 

  

1.2 服务器端阻塞的listenfd

屏蔽以下两行。

 int flags = fcntl(sfd, F_GETFL, 0);                       //获取文件的flags值。
    fcntl(sfd, F_SETFL, flags | O_NONBLOCK);   //设置成非阻塞模式; 

 

阻塞在accept这里。不执行了。 

进程状态如下,一直处于sleep状态中,且不占用cpu.

 2、客户端与服务器连接,但不发数据,测试recv的阻塞情况

客户端程序:

int main()
{

    int cfd;
    int nwrite;
    char *sendbuf="abc";
    sockaddr_in s_sockaddr;
    s_sockaddr.sin_family = AF_INET;
    //字符类型转化为网络序(大端序)
    inet_aton("127.0.0.1",&s_sockaddr.sin_addr);
    s_sockaddr.sin_port = htons(8888);
    cfd = socket(AF_INET,SOCK_STREAM,0);
    if(cfd <0)
    {
        perror("socket:");
        exit(-1);
    }
    if(connect(cfd,(struct sockaddr*)&s_sockaddr,sizeof(s_sockaddr))<0)
    {
        perror("connect:");
        close(cfd);
        exit(-1);
    }
    while(1)
    {
//        nwrite = write(cfd,sendbuf,sizeof(sendbuf));
//        if(nwrite<=0)
//        {
//            perror("write:");
//        }
        sleep(1);
    }


    return 0;
}

2.1服务器端阻塞的cfd

1)此时listenfd为阻塞态;

客户端执行 

 

 

服务器端阻塞在recv上不占用cpu. 

 

 2)此时listenfd为非阻塞态;

 

 小结:服务器端的listenfd要设置阻塞,否则,sleep的秒数不好把控,

一开始设置时间为usleep(10); 客户端一直建立不上连接;并且sleep结束后,此时不一定刚好有客户端过来连接,时间点卡不上。

故listenfd设置为阻塞的较好。

2.2服务器端非阻塞的cfd

由上述可知,我们将listenfd设置为阻塞态,cfd设置为非阻塞态测试。

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<unistd.h>
#include<pthread.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h>
void* ser_fun(void* arg)
{
    pthread_detach(pthread_self());
    long cfd = (long)arg;
    int nread;
    char buf[1024];
    while (1) {
        printf("zhunbei read..\n");
        nread = read(cfd,buf,1024);
        if(nread <=0)
        {
            printf("read:%s,errno=%d\n",strerror(errno),errno);
            usleep(10);
            continue;
        }
        printf("buf=%s\n",buf);
        sleep(1);
    }
}
int main()
{
    int sfd= -1,cfd = -1;
    pthread_t cid;
    struct sockaddr_in s_sockaddr,c_sockaddr;
    int c_sockaddr_len = sizeof(c_sockaddr);//必须赋值
    s_sockaddr.sin_family = AF_INET;
    s_sockaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    s_sockaddr.sin_port = htons(8888);
    sfd = socket(AF_INET,SOCK_STREAM,0);
    if(sfd<0)
    {
        perror("socket:");
        exit(-1);
    }
    int opt=1;
    setsockopt(sfd,SOL_SOCKET, SO_REUSEADDR,
               &opt, sizeof(opt));



    if((bind(sfd,(struct sockaddr*)&s_sockaddr,sizeof(s_sockaddr))<0))//
    {
        perror("bind:");
        close(sfd);
        exit(-1);
    }
    if(listen(sfd,2)<0)
    {
        perror("listen:");
        close(sfd);
        exit(-1);
    }
    while(1)
    {
        if((cfd = accept(sfd,(struct sockaddr*)&c_sockaddr,(socklen_t*)&c_sockaddr_len))<0)
        {
            perror("accept");
            sleep(10);
            continue;
        }
        printf("client link,ip=%s,port= %d,cfd =%d\n",inet_ntoa(c_sockaddr.sin_addr),
               ntohs(c_sockaddr.sin_port),cfd);
        //
        int flags = fcntl(cfd, F_GETFL, 0);                       //获取文件的flags值。
        fcntl(cfd, F_SETFL, flags | O_NONBLOCK);   //设置成非阻塞模式;
        pthread_create(&cid,NULL,ser_fun,(void*)cfd);
    }
    return 0;
}

服务器端打印输出: 

 小结:非阻塞的recv还是消耗cpu的。

总结:

无论时阻塞的cfd还是非阻塞的cfd,只要接收缓冲区内有数据,都会读取并立即返回。

而不论读取到的数据个数是否比希望读取的少。

但当缓冲区中没有数据时,阻塞的cfd会阻塞等待。非阻塞的cfd会返回报错。

send(write)函数的阻塞状态参考该文

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值