ImageVerifierCode 换一换
格式:DOCX , 页数:11 ,大小:148.79KB ,
资源ID:6266010      下载积分:10 金币
快捷注册下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

开通VIP
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.zixin.com.cn/docdown/6266010.html】到电脑端继续下载(重复下载【60天内】不扣币)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

开通VIP折扣优惠下载文档

            查看会员权益                  [ 下载后找不到文档?]

填表反馈(24小时):  下载求助     关注领币    退款申请

开具发票请登录PC端进行申请

   平台协调中心        【在线客服】        免费申请共赢上传

权利声明

1、咨信平台为文档C2C交易模式,即用户上传的文档直接被用户下载,收益归上传人(含作者)所有;本站仅是提供信息存储空间和展示预览,仅对用户上传内容的表现方式做保护处理,对上载内容不做任何修改或编辑。所展示的作品文档包括内容和图片全部来源于网络用户和作者上传投稿,我们不确定上传用户享有完全著作权,根据《信息网络传播权保护条例》,如果侵犯了您的版权、权益或隐私,请联系我们,核实后会尽快下架及时删除,并可随时和客服了解处理情况,尊重保护知识产权我们共同努力。
2、文档的总页数、文档格式和文档大小以系统显示为准(内容中显示的页数不一定正确),网站客服只以系统显示的页数、文件格式、文档大小作为仲裁依据,个别因单元格分列造成显示页码不一将协商解决,平台无法对文档的真实性、完整性、权威性、准确性、专业性及其观点立场做任何保证或承诺,下载前须认真查看,确认无误后再购买,务必慎重购买;若有违法违纪将进行移交司法处理,若涉侵权平台将进行基本处罚并下架。
3、本站所有内容均由用户上传,付费前请自行鉴别,如您付费,意味着您已接受本站规则且自行承担风险,本站不进行额外附加服务,虚拟产品一经售出概不退款(未进行购买下载可退充值款),文档一经付费(服务费)、不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
4、如你看到网页展示的文档有www.zixin.com.cn水印,是因预览和防盗链等技术需要对页面进行转换压缩成图而已,我们并不对上传的文档进行任何编辑或修改,文档下载后都不会有水印标识(原文档上传前个别存留的除外),下载后原文更清晰;试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓;PPT和DOC文档可被视为“模板”,允许上传人保留章节、目录结构的情况下删减部份的内容;PDF文档不管是原文档转换或图片扫描而得,本站不作要求视为允许,下载前可先查看【教您几个在下载文档中可以更好的避免被坑】。
5、本文档所展示的图片、画像、字体、音乐的版权可能需版权方额外授权,请谨慎使用;网站提供的党政主题相关内容(国旗、国徽、党徽--等)目的在于配合国家政策宣传,仅限个人学习分享使用,禁止用于任何广告和商用目的。
6、文档遇到问题,请及时联系平台进行协调解决,联系【微信客服】、【QQ客服】,若有其他问题请点击或扫码反馈【服务填表】;文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“【版权申诉】”,意见反馈和侵权处理邮箱:1219186828@qq.com;也可以拔打客服电话:0574-28810668;投诉电话:18658249818。

注意事项

本文(基本 TCP 套接字编程讲解.docx)为本站上传会员【xrp****65】主动上传,咨信网仅是提供信息存储空间和展示预览,仅对用户上传内容的表现方式做保护处理,对上载内容不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知咨信网(发送邮件至1219186828@qq.com、拔打电话4009-655-100或【 微信客服】、【 QQ客服】),核实后会尽快下架及时删除,并可随时和客服了解处理情况,尊重保护知识产权我们共同努力。
温馨提示:如果因为网速或其他原因下载失败请重新下载,重复下载【60天内】不扣币。 服务填表

基本 TCP 套接字编程讲解.docx

1、基本 TCP 套接字编程讲解 基于 TCP 的套接字编程的所有客户端和服务器端都是从调用socket 开始,它返回一个套接字描述符。客户端随后调用connect 函数,服务器端则调用 bind、listen 和accept 函数。套接字通常使用标准的close 函数关闭,但是也可以使用 shutdown 函数关闭套接字。下面针对套接字编程实现过程中所调用的函数进程分析。以下是基于 TCP 套接字编程的流程图: socket 函数 套接字是通信端点的抽象,实现端对端之间的通信。与应用程序要使用文件描述符访问文件一样,访问套接字需要套接字描述符。任何套接字编程都必须调用soc

2、ket 函数获得套接字描述符,这样才能对套接字进行操作。以下是该函数的描述: 1. /* 套接字 */    2.     3. /*   4.  * 函数功能:创建套接字描述符;   5.  * 返回值:若成功则返回套接字非负描述符,若出错返回-1;   6.  * 函数原型:   7.  */    8. #include     9.     10. int socket(int family, int type, int protocol);    11. /*   12.  * 说明:   13.  * socket类似与open

3、对普通文件操作一样,都是返回描述符,后续的操作都是基于该描述符;   14.  * family 表示套接字的通信域,不同的取值决定了socket的地址类型,其一般取值如下:   15.  * (1)AF_INET         IPv4因特网域   16.  * (2)AF_INET6        IPv6因特网域   17.  * (3)AF_UNIX         Unix域   18.  * (4)AF_ROUTE        路由套接字   19.  * (5)AF_KEY          密钥套接字   20.  * (6)AF_UNSPEC       未

4、指定   21.  *   22.  * type确定socket的类型,常用类型如下:   23.  * (1)SOCK_STREAM     有序、可靠、双向的面向连接字节流套接字   24.  * (2)SOCK_DGRAM      长度固定的、无连接的不可靠数据报套接字   25.  * (3)SOCK_RAW        原始套接字   26.  * (4)SOCK_SEQPACKET  长度固定、有序、可靠的面向连接的有序分组套接字   27.  *   28.  * protocol指定协议,常用取值如下:   29.  * (1)0            

5、   选择type类型对应的默认协议   30.  * (2)IPPROTO_TCP     TCP传输协议   31.  * (3)IPPROTO_UDP     UDP传输协议   32.  * (4)IPPROTO_SCTP    SCTP传输协议   33.  * (5)IPPROTO_TIPC    TIPC传输协议   34.  *   35.  */    connect 函数 在处理面向连接的网络服务时,例如 TCP ,交换数据之前必须在请求的进程套接字和提供服务的进程套接字之间建立连接。TCP 客户端可以调用函数connect 来建立与 TCP 服务器端的一个

6、连接。该函数的描述如下: 1. /*   2.  * 函数功能:建立连接,即客户端使用该函数来建立与服务器的连接;   3.  * 返回值:若成功则返回0,出错则返回-1;   4.  * 函数原型:   5.  */    6. #include     7.     8. int connect(int sockfd, const struct sockaddr *servaddr, socklen_t addrlen);    9. /*   10.  * 说明:   11.  * sockfd是系统调用的套接字描述符,即由socket函

7、数返回的套接字描述符;   12.  * servaddr是目的套接字的地址,该套接字地址结构必须包含目的IP地址和目的端口号,即想与之通信的服务器地址;   13.  * addrlen是目的套接字地址的大小;   14.  *   15.  * 如果sockfd没有绑定到一个地址,connect会给调用者绑定一个默认地址,即内核会确定源IP地址,并选择一个临时端口号作为源端口号;   16.  */    TCP 客户端在调用函数 connect 前不必非得调用 bind 函数,因为内核会确定源 IP 地址,并选择一个临时端口作为源端口号。若 TCP 套接字调用connect

8、函数将建立 TCP 连接(执行三次握手),而且仅在连接建立成功或出错时才返回,其中出错返回可能有以下几种情况: 若 TCP 客户端没有收到 SYN 报文段的响应,则返回 ETIMEOUT 错误; 若客户端的 SYN 报文段的响应是 RST (表示复位),则表明该服务器主机在我们指定的端口上没有进程在等待与之连接。只是一种硬错误,客户端一接收到 RST 就立即返回ECONNERFUSED 错误; RST 是 TCP 在发生错误时发送的一种 TCP 报文段。产生 RST 的三个条件时: 目的地为某端口的 SYN 到达,然而该端口上没有正在监听的服务器; TCP 想取消一个已有连接; T

9、CP 接收到一个不存在的连接上的报文段; 若客户端发出的 SYN 在中某个路由器上引发一个目的地不可达的 ICMP 错误,这是一个软错误。客户端主机内核保存该消息,并在一定的时间间隔继续发送 SYN (即重发)。在某规定的时间后仍未收到响应,则把保存的消息(即 ICMP 错误)作为EHOSTUNREACH 或ENETUNREACH 错误返回给进行。 bind 函数 调用函数 socket 创建套接字描述符时,该套接字描述符是存储在它的协议族空间中,没有具体的地址,要使它与一个地址相关联,可以调用函数bind 使其与地址绑定。客户端的套接字关联的地址一般可由系统默认分配,因此不需要指定具体

10、的地址。若要为服务器端套接字绑定地址,可以通过调用函数 bind 将套接字绑定到一个地址。下面是该函数的描述: 1. /* 套接字的基本操作 */    2.     3. /*   4.  * 函数功能:将协议地址绑定到一个套接字;其中协议地址包含IP地址和端口号;   5.  * 返回值:若成功则返回0,若出错则返回-1;   6.  * 函数原型:   7.  */    8. #include     9. int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen

11、);    10. /*   11.  * 说明:   12.  * sockfd 为套接字描述符;   13.  * addr是一个指向特定协议地址结构的指针;   14.  * addrlen是地址结构的长度;   15.  */    对于 TCP 协议,调用 bind 函数可以指定一个端口号,或指定一个 IP 地址,也可以两者都指定,还可以都不指定。若 TCP 客户端或服务器端不调用bind 函数绑定一个端口号,当调用connect 或 listen 函数时,内核会为相应的套接字选择一个临时端口号。一般 TCP 客户端使用内核为其选择一个临时的端口号,而服务器端通过调用b

12、ind 函数将端口号与相应的套接字绑定。进程可以把一个特定的 IP 地址捆绑到它的套接字上,但是这个 IP 地址必须属于其所在主机的网络接口之一。对于 TCP 客户端,这就为在套接字上发送的 IP 数据报指派了源 IP 地址。对于 TCP 服务器端,这就限定该套接字只接收那些目的地为这个 IP 地址的客户端连接。TCP 客户端一般不把 IP 地址捆绑到它的套接字上。当连接套接字时,内核将根据所用外出网络接口来选择源 IP 地址,而所用外出接口则取决于到达服务器端所需的路径。若 TCP 服务器端没有把 IP 地址捆绑到它的套接字上,内核就把客户端发送的 SYN 的目的 IP 地址作为服务器端的源

13、 IP 地址。 在地址使用方面有下面一些限制: 在进程所运行的机器上,指定的地址必须有效,不能指定其他机器的地址; 地址必须和创建套接字时的地址族所支持的格式相匹配; 端口号必须不小于1024,除非该进程具有相应的特权(超级用户); 一般只有套接字端点能够与地址绑定,尽管有些协议允许多重绑定; listen 函数 在编写服务器程序时需要使用监听函数 listen 。服务器进程不知道要与谁连接,因此,它不会主动地要求与某个进程连接,只是一直监听是否有其他客户进程与之连接,然后响应该连接请求,并对它做出处理,一个服务进程可以同时处理多个客户进程的连接。listen 函数描述如下:

14、1. /*   2.  * 函数功能:接收连接请求;   3.  * 函数原型:   4.  */    5. #include     6.     7. int listen(int sockfd, int backlog);//若成功则返回0,若出错则返回-1;    8. /*   9.  * sockfd是套接字描述符;   10.  * backlog是该进程所要入队请求的最大请求数量;   11.  */    listen 函数仅由 TCP 服务器调用,它有以下两种作用: 当 socket 函数创建一个套接字时,若它被假设为一

15、个主动套接字,即它是一个将调用connect 发起连接的客户端套接字。listen 函数把一个未连接的套接字转换成一个被动套接字,指示内核应该接受指向该套接字的连接请求; listen 函数的第二个参数规定内核应该为相应套接字排队的最大连接个数; listen 函数一般应该在调用socket 和bind 这两个函数之后,并在调用 accept 函数之前调用。 内核为任何一个给定监听套接字维护两个队列: 未完成连接队列,每个这样的 SYN 报文段对应其中一项:已由某个客户端发出并到达服务器,而服务器正在等待完成相应的 TCP 三次握手过程。这些套接字处于 SYN_REVD 状态; 已完成

16、连接队列,每个已完成 TCP 三次握手过程的客户端对应其中一项。这些套接字处于 ESTABLISHED 状态; accept 函数 accept 函数由 TCP 服务器调用,用于从已完成连接队列队头返回下一个已完成连接。如果已完成连接队列为空,那么进程被投入睡眠。该函数的返回值是一个新的套接字描述符,返回值是表示已连接的套接字描述符,而第一个参数是服务器监听套接字描述符。一个服务器通常仅仅创建一个监听套接字,它在该服务器的生命周期内一直存在。内核为每个由服务器进程接受的客户连接创建一个已连接套接字(表示 TCP 三次握手已完成),当服务器完成对某个给定客户的服务时,相应的已连接套接字就会被

17、关闭。该函数描述如下: 1. /* 函数功能:从已完成连接队列队头返回下一个已完成连接;若已完成连接队列为空,则进程进入睡眠;   2.  * 函数原型:   3.  */    4. int accept(int sockfd, struct sockaddr *cliaddr, socklen_t *addrlen);//返回值:若成功返回套接字描述符,出错返回-1;    5. /*   6.  * 说明:   7.  * 参数 cliaddr 和 addrlen 用来返回已连接的对端(客户端)的协议地址;   8.  *   9.  * 该函数返回套接字描述符,该描述

18、符连接到调用connect函数的客户端;   10.  * 这个新的套接字描述符和原始的套接字描述符sockfd具有相同的套接字类型和地址族,而传给accept函数的套接字描述符sockfd没有关联到这个链接,   11.  * 而是继续保持可用状态并接受其他连接请求;   12.  * 若不关心客户端协议地址,可将cliaddr和addrlen参数设置为NULL,否则,在调用accept之前,应将参数cliaddr设为足够大的缓冲区来存放地址,   13.  * 并且将addrlen设为指向代表这个缓冲区大小的整数指针;   14.  * accept函数返回时,会在缓冲区填充客户

19、端的地址并更新addrlen所指向的整数为该地址的实际大小;   15.  *   16.  * 若没有连接请求等待处理,accept会阻塞直到一个请求到来;   fork 和 exec 函数 1. /* 函数功能:创建子进程;   2.  * 返回值:   3.  * (1)在子进程中,返回0;   4.  * (2)在父进程中,返回新创建子进程的进程ID;   5.  * (3)若出错,则范回-1;   6.  * 函数原型:   7.  */    8. #include     9. pid_t fork(void);    10. /* 

20、说明:   11.  * 该函数调用一次若成功则返回两个值:   12.  * 在调用进程(即父进程)中,返回新创建进程(即子进程)的进程ID;   13.  * 在子进程返回值是0;   14.  * 因此,可以根据返回值判断进程是子进程还是父进程;   15.  */    16.     17. /* exec 序列函数 */    18.     19. /*   20.  * 函数功能:把当前进程替换为一个新的进程,新进程与原进程ID相同;   21.  * 返回值:若出错则返回-1,若成功则不返回;   22.  * 函数原型:   23.  */   

21、 24. #include     25. int execl(const char *pathname, const char *arg, ...);    26. int execv(const char *pathnam, char *const argv[]);    27. int execle(const char *pathname, const char *arg, ... , char *const envp[]);    28. int execve(const char *pathnam, char *const argv[], char *c

22、onst envp[]);    29. int execlp(const char *filename, const char *arg, ...);    30. int execvp(const char *filename, char *const argv[]);    31. /*  6 个函数的区别如下:   32.  * (1)待执行的程序文件是 文件名 还是由 路径名 指定;   33.  * (2)新程序的参数是 一一列出 还是由一个 指针数组 来引用;   34.  * (3)把调用进程的环境传递给新程序 还是 给新程序指定新的环境;   35.  */  

23、  exec 6个函数在函数名和使用语法的规则上都有细微的区别,下面就从可执行文件查找方式、参数传递方式及环境变量这几个方面进行比较。 查找方式:前4个函数的查找方式都是完整的文件目录路径 pathname ,而最后两个函数(也就是以p结尾的两个函数)可以只给出文件名 filename,系统就会自动按照环境变量 “$PATH” 所指定的路径进行查找。 参数传递方式:exec 序列函数的参数传递有两种方式:一种是逐个列举的方式,而另一种则是将所有参数整体构造指针数组传递。在这里是以函数名的第5位字母来区分的,字母为 “l”(list)的表示逐个列举参数的方式,其语法为 const char

24、 *arg;字母为 “v”(vertor)的表示将所有参数整体构造指针数组传递,其语法为 char *const argv[]。读者可以观察 execl()、execle()、execlp() 的语法与 execv()、execve()、execvp() 的区别。这里的参数实际上就是用户在使用这个可执行文件时所需的全部命令选项字符串(包括该可执行程序命令本身)。要注意的是,这些参数必须以NULL结束。 环境变量:exec 序列函数可以默认系统的环境变量,也可以传入指定的环境变量。这里以 “e”(environment)结尾的两个函数 execle() 和 execve() 就可以在 envp

25、[] 中指定当前进程所使用的环境变量。 1.  表 1 exec 序列函数的总结        2. 前4位 统一为:exec        3. 第5位 l:参数传递为逐个列举方式    execl、execle、execlp    4.      v:参数传递为构造指针数组方式     execv、execve、execvp    5. 第6位 e:可传递新进程环境变量     execle、execve    6.      p:可执行文件查找方式为文件名     execlp、execvp    其关系如下图: 并发服务器 当要求一个服务器同时为多个客户服务时,需

26、要并发服务器。TCP 并发服务器,它们为每个待处理的客户端连接调用 fork 函数派生一个子进程。当一个连接建立时,accept 返回,服务器接着调用 fork 函数,然后由子进程服务客户端,父进程则等待另一个连接,此时,父进程必须关闭已连接套接字。 close 和 shutdown 函数 当要关闭套接字时,可使用 close 和 shutdown 函数,其描述如下: 1. /* 函数功能:关闭套接字,若是在 TCP 协议中,并终止 TCP 连接;   2.  * 返回值:若成功则返回0,若出错则返回-1;   3.  * 函数原型:   4.  */    5. #includ

27、e     6. int close(int sockfd);    7.     8. /*   9.  * 函数功能:关闭套接字上的输入或输出;   10.  * 返回值:若成功则返回0,若出错返回-1;   11.  * 函数原型:   12.  */    13. #include     14. int shutdown(int sockfd, int how);    15. /*   16.  * 说明:   17.  * sockfd表示待操作的套接字描述符;   18.  * how表示具体操作,取值如

28、下:   19.  * (1)SHUT_RD     关闭读端,即不能接收数据   20.  * (2)SHUT_WR     关闭写端,即不能发送数据   21.  * (3)SHUT_RDWR   关闭读、写端,即不能发送和接收数据   22.  *   23.  */    getsockname 和 getpeername 函数 为了获取已绑定到套接字的地址,我们可以调用函数 getsockname 来实现: 1. /*   2.  * 函数功能:获取已绑定到一个套接字的地址;   3.  * 返回值:若成功则返回0,若出错则返回-1;   4.  * 函数原型:

29、   5.  */    6. #include     7.     8. int getsockname(int sockfd, struct sockaddr *addr, socklen_t *alenp);    9. /*   10.  * 说明:   11.  * 调用该函数之前,设置alenp为一个指向整数的指针,该整数指定缓冲区sockaddr的大小;   12.  * 返回时,该整数会被设置成返回地址的大小,如果该地址和提供的缓冲区长度不匹配,则将其截断而不报错;   13.  */    14. /*   15.  * 函

30、数功能:获取套接字对方连接的地址;   16.  * 返回值:若成功则返回0,若出错则返回-1;   17.  * 函数原型:   18.  */    19. #include     20.     21. int getpeername(int sockfd, struct sockaddr *addr, socklen_t *alenp);    22. /*   23.  * 说明:   24.  * 该函数除了返回对方的地址之外,其他功能和getsockname一样;   25.  */    【编辑推荐】 1. 图解TCP-IP协议 2. TCP 与 UDP 协议基本概述 3. 解读TCP 四种定时器

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        抽奖活动

©2010-2026 宁波自信网络信息技术有限公司  版权所有

客服电话:0574-28810668  投诉电话:18658249818

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :微信公众号    抖音    微博    LOFTER 

客服