范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

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

  问题引出
  好几个读者私信说在腾讯面试过程中,被面试官问到了一个问题:"一个tcp服务端和一个tcp客户端,客户端和服务端建立连接后,服务端一直sleep,然后客户端一直发送数据会是什么现象"。
  要回答这个问题,需要我们清楚tcp协议的特点和tcp发送数据的大体过程。
  tcp发送数据过程
  恐怕接触过网络的同学都知道tcp是面向连接的可靠传输协议,意味着客户端发送的数据服务端是一定能够收到的,那么对于上面的问题就不可能存在数据的丢弃。下面我们分析一下tcp的传输过程。
  如图所示,tcp数据包的传输过程主要有如下几个步骤:
  • 1.应用程序调用write系列函数发送数据 ,数据首先由应用程序缓冲区复制到发送端的内核中的 套接字发送缓冲区,然后write成功返回;需要特别注意的是write成功返回只是说明数据成功的由应用进程缓冲区复制到了套接字发送缓冲区,并不代表数据发送到了对端主机。• 2.内核协议栈将套接字发送缓冲区中的数据发送到对端主机,这个过程不受应用程序控制,而是发送端内核协议栈完成;• 3.数据到达接收端主机的套接字接收缓冲区,注意这个接收过程也不受应用程序控制,而是由接收端内核协议栈完成;• 4.数据由套接字接收缓冲区复制到接收端应用程序缓冲区,注意这个过程是由类似read等函数来完成。
  清楚了tcp的传输过程,现在我们分情况来讨论上面的问题。
  相关视频推荐
  从websocket协议到tcp自定义协议,tcp分包与粘包
  90分钟搞定tcp/ip协议栈
  TCPIP协议栈,一次课开启你的网络之门
  LinuxC++后台服务器开发架构师免费学习地址:C/C++Linux鏈嶅姟鍣ㄥ紑鍙 /鍚庡彴鏋舵瀯甯堛 愰浂澹版暀鑲层 -瀛︿範瑙嗛 鏁欑 -鑵捐 璇惧爞
  【文章福利】:小编整理了一些个人觉得比较好的学习书籍、视频资料共享在群文件里面,有需要的可以自行添加哦!(需要视频资料后台私信"1"自取)
  阻塞方式的情况
  write系列函数的工作方式默认是阻塞方式:调用write函数时,内核从应用进程的缓冲区到套接字的发送缓冲区复制数据。如果其发送缓冲区中没有空间,进程将进入睡眠,直到有空间为止。
  因此,阻塞方式下,如果服务端一直sleep不接收数据,而客户端一直write,也就是只能执行上述过程中的前三步,这样最后接收端的套接字接收缓冲区和发送端套接字发送缓冲区都被填满,这样write就无法继续将数据从应用程序复制到发送端的套接字发送缓冲区了,从而发送端进程进入睡眠。可以用下面的程序验证。
  tcpClient.c是客户端代码用来发送数据,客户端每次write成功一次,将计数器count加1,同时输出本次write成功的字节数。count保存客户端write成功的次数。
  #include  #include  #include  #include  #include  #include  #include  #include  #include  #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,count);     }      if(n<0)         perror("write error");     close(sockfd); }
  下面的tcpServer.c是服务端程序,服务端并不接收数据。
  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #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 ");         if((pid = fork()) == 0)         {             close(listenfd);             sleep(1000);//子进程不接收数据,sleep 1000秒             exit(0);         }         else         {             close(connfd);         }     } }
  首先编译运行服务端,然后启动客户端,运行结果如下:
  可以看到客户端write成功377次后就陷入了阻塞,注意这个时候不能说明发送端的套接字发送缓冲区一定是满的,只能说明套接字发送缓冲区的可用空间小于write请求写的自己数——1024。
  非阻塞方式的情况
  下面看一下非阻塞套接字情况下,write的工作方式:对于一个非阻塞的TCP套接字,如果发送缓冲区中根本没用空间,输出函数将立即返回一个EWOULDBLOCK错误。如果发送缓冲区中有一些空间,返回值将是内核能够复制到该缓冲区的字节数。这个字节数也成为"不足计数"。
  这样就可以知道非阻塞情况下服务端一直sleep,客户端一直write数据的效果了:开始客户端write成功,随着客户端write,接收端的套接字接收缓冲区和发送端的套接字发送缓冲区会被填满。当发送端的套接字发送缓冲区的可用空间小于write请求写的字节数时,write立即返回-1,并将errno置为EWOULDBLOCK。
  可以用下面的程序验证,其中,服务端程序代码和上面例子一样,我们只看客户端非阻塞模式代码:
  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #include  #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,count);    }         if(n<0)   {     if(errno!=EWOULDBLOCK)        perror("write error");     else        printf("EWOULDBLOCK ERROR ");    }    close(sockfd); }
  首先编译运行服务端,然后启动客户端,运行结果如下图所示。
  编辑
  可以看到客户端成功write 185次后就发生套接字发送缓冲区空间不足,从而返回EWOULDBLOCK错误。我们注意到每次write同样的字节数(1024)阻塞模式下能write成功377次,为什么非阻塞情况下要少呢?
  这是因为阻塞模式下一直write到接收端的套接字接收缓冲区和发送端的套接字发送缓冲区都满的情况才会阻塞。而非阻塞模式情况下有可能是发送端发送过程的第二步较慢,造成发送端的套接字发送缓冲区很快写满,而接收端的套接字接收缓冲区还没有满,这样write就会仅仅因为发送端的套接字发送缓冲区满而返回错误。
  原文地址:https://mp.weixin.qq.com/s/rpNTjTUt19Bbyx6IWm2-ig

为什么长城的尽头在大海里?老外看后大呼中国人太聪明了先问大家一个小问题,如果你是一个外国人,有人向你提到中国的话,那么你首先想到的是什么?当然,很多人首先想到的就是万里长城,万里长城也是中国在国际上的一个符号,相信大家就更熟悉了。很历史上10月27日都发生了什么?1275年阿姆斯特丹建城阿姆斯特丹(Amsterdam),荷兰首都及最大城市,位于荷兰西部的北荷兰省。享誉世界的旅游城市和国际大都市,在最新的世界城市排名权威机构GaWC中名列欧洲入住酒店时,发现浴室都爱用透明玻璃,不会感到别扭吗?在到酒店入住的时候,相信大家都会见到透明玻璃的浴室。那么酒店的浴室设计成透明状,顾客们会不会感到别扭呢?(此处已添加小程序,请到今日头条客户端查看)其实酒店的浴室之所以采用透明玻璃混双单打双丰收!乒乓全青赛大结局男单向鹏夺冠,女单蒯曼卫冕今天(10月27日),是2021全国青年乒乓球锦标赛最后一天,男女乒单打冠军将产生。最终1)男乒方面向鹏41战胜上届冠军林诗栋,摘得男单桂冠,从而,他个人这次也包揽了团体混双单打三美媒预测最佳新秀!状元郎排第二,3战15分后卫上榜,火箭赚大了随着新赛季的开始,2021届选秀球员即将登陆NBA,迎来自己职业生涯的第一场常规赛,21届选秀大会可以说是人才辈出,又被称为选秀一大年,这些新秀们更是被认为有多人将会称为未来联盟门首届总台中欧音乐节暨北京冬奥会倒计时100天音乐会成功举办在北京冬奥会倒计时100天到来之际,中央广播电视总台成功举办中欧音乐节暨北京冬奥会倒计时100天音乐会。来自欧洲多国的百余名音乐家以更团结共筑梦的激情序曲,共同祝福北京2022冬奥NBA战报湖人爵士连胜,76人火箭输球,勇士联盟第一创36年纪录文聪古说球NBA今天安排了5场比赛,湖人和爵士纷纷取得连胜,76人和火箭队输球,勇士队开季豪取4连胜,与公牛战绩并列联盟第一。并创造了一个近36年的纪录揭幕战至今的4场比赛,勇士在女排世俱杯6参赛队出炉,意土3强合围冠军?10月26日,2021年南美女排俱乐部锦标赛落幕。巴西乌兰贝迪亚海滩俱乐部夺冠后,获得今年世俱杯资格,世俱杯的六支参赛球队至此全部出炉。今年南美女排俱乐部锦标赛在巴西首都巴西利亚举何炅的哭泣,撕开了娱乐圈那层华丽的外衣何炅哭了,而且是痛哭!近日,在浙江乌镇,何炅一改往日温文尔雅的形象,躺在周迅怀里痛哭。熟悉何炅的人都知道,何炅是个爱哭的人。好友黄磊就曾经调侃何炅何老师非常爱哭的,他有主动哭,还有这6位女演员,既无演技又无颜值,关系户三个字仿佛直接写在脸上在现如今这个什么都讲关系的社会,喜欢走捷径的关系户人员在各个领域行业都很常见。这些人技术能力堪忧,却往往占据一大把好资源,普通人拿她们真的是无可奈何,吃了亏也只能哑巴吃黄连有口说不爱情公寓男演员现状2人娶了搭档,2人暴瘦20斤,1人出国读硕士之前发布了爱情公寓女演员现状3人沉迷直播带货,1人离婚,1人淡出娱乐圈,今天就来细数一下爱情公寓男演员的现状2人娶了搭档,2人暴瘦20斤,1人出国读了硕士。1陈赫陈赫凭借着爱情公寓
vivo手机哪款性价比高质量好?别再说被坑了!其实高手在入这几款近年来,vivo品牌一改高价低配的印象,推出超高性价比全新独立子品牌iQOO,其产品与vivo品牌自有工厂同一生产线,保证的质量与高性价比的同时,更是力压为发烧而生的小米家族。如今王慧文CopyOpenAItoChina一条朋友圈,一个美团联合创始人身份,将ChatGPT的火热推到了一个全新高度。前几天,王慧文拿出5000万美元向天下英雄招榜,随后真格源码两大资本2。3亿美元光速入局,令整个科技圈案例分享从光明到坪山,贝特瑞集团如何实现跨区智慧办公?贝特瑞集团是行业地位突出的新能源材料研发与制造商。光明区贝特瑞集团总部大厦位于光明区,主要用于集团总部办公,占地面积7000平方米,地上主体建筑31层。大厦涵盖员工食堂,企业展层,微软聊天机器人被指辱骂用户?它是怎么回答的头条创作挑战赛必应人工智能(BingAI)是指微软在必应搜索引擎和Edge浏览器中整合的一种人工智能聊天机器人功能,采用微软自研的ChatGPT模型,可以根据用户的查询内容和意图提不让汉DM,汉EV专美于前!岚图追光即将上市,起步510PS马力!新能源汽车市场拥有更快的变化速度,原本中大型款式作为豪华领域,车型更新频率属于较慢的类型,受众限制清晰,购买难度较大,但随电驱款式普及,使用与养护成本大幅度抑制,使更多消费者开始渐理想L系OTA4。3即将推送新增35项新功能优化17项体验爱卡汽车行业资讯原创理想汽车在2月17日宣布,理想L8与L9车型的OTA4。3即将正式推送,官方将分批次陆续完成推送。4。3系统中增加了35项新功能,优化17项体验。本次OTA更新苹果当日达服务遭投诉外卖员窃取iPhone后,用户退款困难IT之家2月17日消息,苹果美国的很多AppleStore零售店都提供2小时送达服务,费用为9美元。不过在过去几年时间里,这项服务遭到了很多用户的投诉,最大的两个问题就是产品被盗和苹果用户称ID被盗致微信遭盗刷近三万元,两次申请退款未通过近日,家住重庆的黄先生向澎湃质量观反映,2月9日他的苹果手机突然关机,开机后原有AppleID号无法登录,他立即与苹果客服沟通。在沟通过程中,账户资金被盗刷22笔,每笔1298元,消息称NAND芯片价格下跌1TB将成旗舰手机标准规格手机中国新闻目前智能手机上面的应用越来越多,占据的内存也越来越大,以往的所谓256GB已经称不上是大内存了,消费者在预算足够的情况下会更愿意选择512GB版本,甚至直接选择1TB版安卓用户心头爱,全网资源免费下,纯净无广不知道昨天的情人节有多少朋友是出去约会的,虽然小雷没过节,但实实在在体验到了节日的氛围,下班的时候地铁都挤不上去,实属是一把辛酸泪。虽然没人陪,但一个人有一个人的快乐,在家看看电影亚马逊速卖通成为全球最受欢迎两大跨境电商平台2月14日最新跨境电商调研报告显示,亚马逊和阿里巴巴速卖通是全球消费者最爱使用的两大跨境电商平台。调研覆盖了来自39个国家的共3。3万名消费者。当被问及他们最近一次使用的跨境电商平