一道高频腾讯面试题:tcp数据发送问题

2022年07月15日20:52:27 科技 1165


问题引出


好几个读者私信说在腾讯面试过程中,被面试官问到了一个问题:“一个tcp服务端和一个tcp客户端,客户端和服务端建立连接后,服务端一直sleep,然后客户端一直发送数据会是什么现象”。


要回答这个问题,需要我们清楚tcp协议的特点和tcp发送数据的大体过程。




tcp发送数据过程


恐怕接触过网络的同学都知道tcp是面向连接的可靠传输协议,意味着客户端发送的数据服务端是一定能够收到的,那么对于上面的问题就不可能存在数据的丢弃。下面我们分析一下tcp的传输过程。


一道高频腾讯面试题:tcp数据发送问题 - 天天要闻



如图所示,tcp数据包的传输过程主要有如下几个步骤:


  • • 1.应用程序调用write系列函数发送数据 ,数据首先由应用程序缓冲区复制到发送端的内核中的 套接字发送缓冲区,然后write成功返回;需要特别注意的是write成功返回只是说明数据成功的由应用进程缓冲区复制到了套接字发送缓冲区,并不代表数据发送到了对端主机。
  • • 2.内核协议栈将套接字发送缓冲区中的数据发送到对端主机,这个过程不受应用程序控制,而是发送端内核协议栈完成;
  • • 3.数据到达接收端主机的套接字接收缓冲区,注意这个接收过程也不受应用程序控制,而是由接收端内核协议栈完成;
  • • 4.数据由套接字接收缓冲区复制到接收端应用程序缓冲区,注意这个过程是由类似read等函数来完成。


清楚了tcp的传输过程,现在我们分情况来讨论上面的问题。




相关视频推荐

一道高频腾讯面试题:tcp数据发送问题 - 天天要闻




阻塞方式的情况


write系列函数的工作方式默认是阻塞方式:调用write函数时,内核从应用进程的缓冲区到套接字的发送缓冲区复制数据。如果其发送缓冲区中没有空间,进程将进入睡眠,直到有空间为止。


因此,阻塞方式下,如果服务端一直sleep不接收数据,而客户端一直write,也就是只能执行上述过程中的前三步,这样最后接收端的套接字接收缓冲区和发送端套接字发送缓冲区都被填满,这样write就无法继续将数据从应用程序复制到发送端的套接字发送缓冲区了,从而发送端进程进入睡眠。可以用下面的程序验证。


tcpClient.c是客户端代码用来发送数据,客户端每次write成功一次,将计数器count加1,同时输出本次write成功的字节数。count保存客户端write成功的次数。



#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <memory.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#define PORT 8888
#define Buflen 1024
int main(int argc,char *argv[])
{
    struct sockaddr_in server_addr;
    int n,count=0;
    int sockfd;
    char sendline[Buflen];
    sockfd= socket(AF_INET,SOCK_STREAM,0);
    memset(&server_addr,0,sizeof(server_addr));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(PORT);
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
    server_addr.sin_addr.s_addr = inet_addr(argv[1]);
    connect(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr));

    //与服务器端进行通信
    memset(sendline,'x',sizeof(Buflen));

    while ( (n=write(sockfd,sendline,Buflen))>0 )
    {
        count++;
        printf("already write %d bytes -- %d\n",n,count);
    }

    if(n<0)
        perror("write error");
    close(sockfd);
}



下面的tcpServer.c是服务端程序,服务端并不接收数据。



#include <stdio.h>
#include <stdlib.h>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <memory.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#define PORT 8888 //定义通信端口
#define BACKLOG 5 //定义侦听队列长度
#define buflen 1024
int listenfd,connfd;
int main(int argc,char *argv[])
{
    struct sockaddr_in server_addr; //存储服务器端socket地址结构
    struct sockaddr_in client_addr; //存储客户端 socket地址结构
    pid_t pid;
    listenfd = socket(AF_INET,SOCK_STREAM,0);
    memset(&server_addr,0,sizeof(server_addr));
    server_addr.sin_family = AF_INET; //协议族
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY); //本地地址
    server_addr.sin_port = htons(PORT);
    bind(listenfd,(struct sockaddr *)&server_addr,sizeof(server_addr));
    listen(listenfd,BACKLOG);
    for(;;)
    {
        socklen_t addrlen = sizeof(client_addr);
        connfd = accept(listenfd,(struct sockaddr *)&client_addr,&addrlen);
        if(connfd<0)
            perror("accept error");
        printf("receive connection\n");
        if((pid = fork()) == 0)
        {
            close(listenfd);
            sleep(1000);//子进程不接收数据,sleep 1000秒
            exit(0);
        }
        else
        {
            close(connfd);
        }
    }
}



首先编译运行服务端,然后启动客户端,运行结果如下:


一道高频腾讯面试题:tcp数据发送问题 - 天天要闻



可以看到客户端write成功377次后就陷入了阻塞,注意这个时候不能说明发送端的套接字发送缓冲区一定是满的,只能说明套接字发送缓冲区的可用空间小于write请求写的自己数——1024。


非阻塞方式的情况


下面看一下非阻塞套接字情况下,write的工作方式:对于一个非阻塞的TCP套接字,如果发送缓冲区中根本没用空间,输出函数将立即返回一个EWOULDBLOCK错误。如果发送缓冲区中有一些空间,返回值将是内核能够复制到该缓冲区的字节数。这个字节数也成为“不足计数”。


这样就可以知道非阻塞情况下服务端一直sleep,客户端一直write数据的效果了:开始客户端write成功,随着客户端write,接收端的套接字接收缓冲区和发送端的套接字发送缓冲区会被填满。当发送端的套接字发送缓冲区的可用空间小于write请求写的字节数时,write立即返回-1,并将errno置为EWOULDBLOCK。


可以用下面的程序验证,其中,服务端程序代码和上面例子一样,我们只看客户端非阻塞模式代码:



#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <memory.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <fcntl.h>
#include <errno.h>
#define PORT 8888
#define Buflen 1024
int main(int argc,char *argv[])
{
    struct sockaddr_in server_addr;
    int n,flags,count=0;
    int sockfd;
    char sendline[Buflen];
    sockfd= socket(AF_INET,SOCK_STREAM,0);
    memset(&server_addr,0,sizeof(server_addr));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(PORT);
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
    server_addr.sin_addr.s_addr = inet_addr(argv[1]);
    connect(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr));
    flags=fcntl(sockfd,F_GETFL,0); //将已连接的套接字设置为非阻塞模式
    fcntl(sockfd,F_SETFL,flags|O_NONBLOCK);
    memset(sendline,'a',sizeof(Buflen));
    
    while ( (n=write(sockfd,sendline,Buflen))>0 )
   {
     count++;
     printf("already write %d bytes -- %d\n",n,count);
   }
    
   if(n<0)
  {
    if(errno!=EWOULDBLOCK)
       perror("write error");
    else
       printf("EWOULDBLOCK ERROR\n"); 
  }
   close(sockfd);
}


一道高频腾讯面试题:tcp数据发送问题 - 天天要闻


首先编译运行服务端,然后启动客户端,运行结果如下图所示。


一道高频腾讯面试题:tcp数据发送问题 - 天天要闻


一道高频腾讯面试题:tcp数据发送问题 - 天天要闻

编辑


可以看到客户端成功write 185次后就发生套接字发送缓冲区空间不足,从而返回EWOULDBLOCK错误。我们注意到每次write同样的字节数(1024)阻塞模式下能write成功377次,为什么非阻塞情况下要少呢?


这是因为阻塞模式下一直write到接收端的套接字接收缓冲区和发送端的套接字发送缓冲区都满的情况才会阻塞。而非阻塞模式情况下有可能是发送端发送过程的第二步较慢,造成发送端的套接字发送缓冲区很快写满,而接收端的套接字接收缓冲区还没有满,这样write就会仅仅因为发送端的套接字发送缓冲区满而返回错误。


原文地址:https://mp.weixin.qq.com/s/rpNTjTUt19Bbyx6IWm2-ig

科技分类资讯推荐

8月正式发布,小鹏MONA M03开启全球首秀 - 天天要闻

8月正式发布,小鹏MONA M03开启全球首秀

量化之美,破风而来。7月3日下午14点,小鹏MONA M03开启全球首秀。这款为年轻用户打造的智能纯电掀背轿跑,以其独特的AI量化美学设计吸引了行业关注。小鹏汽车董事长CEO何小鹏携手造型中心副总裁胡安马·洛佩兹(JuanMa Lopez)共同出席了此次直播,深入解读了小鹏MONA M03的设计创作理念和背后的技术实力。AI量化美学设计...
外媒:马斯克或线上出席世界人工智能大会 人形机器人Optimus也将参展 - 天天要闻

外媒:马斯克或线上出席世界人工智能大会 人形机器人Optimus也将参展

腾讯科技讯 7月3日消息,据国外媒体报道,埃隆·马斯克(Elon Musk)拟于周四在上海举办的世界人工智能大会(WAIC)开幕式上发表主旨演讲。尽管议程未明确指出马斯克是现场出席活动还是通过视频连线方式参与,但据知情人士透露,他预计不会亲自到场。对于关于马斯克出席方式及演讲内容的询问,WAIC的组织者尚未给予回应,特...
苹果Vision Pro市场表现“温和”  推动国内市场高端化 - 天天要闻

苹果Vision Pro市场表现“温和” 推动国内市场高端化

21世纪经济报道记者倪雨晴 实习生刘冰仪、朱梓烨 深圳、广州报道近日,苹果Vision Pro国行版正式发售。与今年1月美国预售时迅速售罄、导致发货延迟至三月中旬的热烈反响相比,国行版的Vision Pro市场表现显得较为“温和”。
LG推出全球首款144Hz无线4K电视 - 天天要闻

LG推出全球首款144Hz无线4K电视

LG公司LG OLED evo M4 是第一款能够接收无线传输的视频和音频的电视,分辨率为 4K,刷新率为 144 Hz。为此,该套件包括一个连接盒,可让您在电视和接收器之间建立空中连接。
家用优选全能SUV博越L,时尚科技并存,理想出行伙伴! - 天天要闻

家用优选全能SUV博越L,时尚科技并存,理想出行伙伴!

在紧凑型SUV这片竞争激烈的红海中,一款车型想要脱颖而出,不仅需要在某个方面做到极致,更需具备全方位的均衡实力。吉利新博越L正是这样一款车型,它以卓越的设计美学、前瞻的智能科技、灵活实用的空间布局,以及高效的动力系统,精准击中了现代家庭用车的多元需求,被誉为“
小米SU7导航信号差,真让余承东说中了,雷军再三解释也没用 - 天天要闻

小米SU7导航信号差,真让余承东说中了,雷军再三解释也没用

谁也没想到余承东的一席话,让小米车主的心凉了半截。最近,小米su7导航不准信号差的问题冲上了热搜,正当车主们一筹莫展的时候,余承东在新问界M5发布会上说的实话,给大家揭晓了答案。那时,余承东先是坦白了自己做无线电出身的事实,然后直言汽车前风
618购车狂欢,不能错过吉利星越L - 天天要闻

618购车狂欢,不能错过吉利星越L

​天猫年中大促进行时,就连汽车行业也来了一场618购车盛宴。吉利就拿出了它的明星产品——这个集高科技、豪华配置于一身的星越L,让每一位向往品质生活的消费者都无法抗拒。