本篇文章介绍TCP通信。

上文提到传输层的两个协议TCP和UDP,UDP是无连接的已经介绍过,TCP是面向连接的,阐述建立连接和断开连接前先来看下TCP报文头的结构。

报文头在linux的定义在/usr/include/netinet/tcp.h中:

struct tcphdr
{
u_int16_t source; //源端口号
u_int16_t dest; //目的端口号
u_int32_t seq; //32位的TCP报文序列号
u_int32_t ack_seq; //32位的TCP报文确认序列号
u_int16_t res1:4; //保留位
u_int16_t doff:4; //首部长度
u_int16_t fin:1; //fin置1表示该报文用于申请断开TCP连接
u_int16_t syn:1; //syn置1表示该报文用于申请建立TCP连接
u_int16_t rst:1; //rst置1表示该报文用于申请重建TCP连接
u_int16_t psh:1; //psh置1表示该报文的优先级较高(用于发送紧急报文)
u_int16_t ack:1; //ack置1表示该报文具有确认的功能,此时确认序列号有效
u_int16_t urg:1; //urg置1使紧急指针有效
u_int16_t res2:2; //保留位(加上res1共6位)
u_int16_t window; //窗口大小,用于流量控制
u_int16_t check; //tcp报文的校验和
u_int16_t urg_ptr; //紧急指针(是一个偏移量),序列号到紧急指针之间的数据为紧急数据,紧急指针后的数据才是正常数据
};

可以看出来,TCP报文首部设计的功能要比UDP报文首部复杂的多(可靠自然带来大量的额外开销),在建立连接中我们比较关心的是32位的序列号、32位的确认序列号、SYN位、ACK位,通过下面的图形我们来看下TCP建立连接的三次握手过程(三次握手指的是三次报文的传输),其中发起连接的一端我们称为主动端,等待连接的一端我们称为被动端。

第一次握手:主动端向被动端发送一个syn置1,序列号为x的一个tcp报文

第二次握手:被动端向主动端回溯一个ack置1,确认序列号为x+1的tcp报文同时将该报文的syn置1并生成一个序列号y,以此使该报文又具有了发起连接的功能

第三次握手:主动端再向被动端回溯一个ack置1,确认序列号为y+1的的确认报文,到此完成了tcp连接的建立。

简单总结下:完成tcp建立连接需要两端都进行以此连接申请和申请的确认,但总会有一个主动端先申请建立连接。下面我们来看下如何通过函数调用来完成整个建立的阶段。

先来看下被动端,有一个阶段是等待连接请求的阶段,通过listen()函数开启连接的监听等待:

int listen(int sockfd, int backlog);

参数sockfd是在哪个套接字上实现监听,backlog是最多能够建立几个连接(已经完成三次握手的)。listen()函数并不会阻塞等待,而是开启监听等待,开启后的等待过程是由内核的协议栈完成的,此时sockfd套接字已成为监听描述符,该描述符的可读条件变成有连接完成,当有连接完成时通过accept()函数来读出完成的连接并返回该TCP套接字描述符:

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);

参数sockfd是监听描述符,addr是主动端的网络地址,addrlen是地址长度。该函数被调用后会阻塞至sockfd可读(即有连接完成),返回tcp套接字描述符。注意在TCP的被动端至少有两个套接字描述符,一个是监听描述符(用socket()创建并用listen开启监听状态),一个是tcp通信的描述符(由内核创建并由accept返回)。

连接建立之后,套接字的通信双方已经确定,在通信时就不必对方的网络地址,直接使用read()/write()传输数据即可。

TCP连接的被动端的函数调用过程:socket()创建监听描述符-->bind()绑定本地网络地址-->listen()开启监听状态-->accept()获得建立的tcp连接的套接字描述符-->read()/write()进行数据通信-->close()关闭套接字(监听描述符结束监听状态,tcp连接描述符断开TCP连接)。

在来说下TCP的主动端,主动端使用connect()函数发起tcp连接的建立:

int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

参数sockfd是套接字描述符(由socket()创建),addr是被动端的网络地址,addrlen是addr的长度。该函数将阻塞至内核完成三次握手,主动端只有一个描述符(无监听描述符)。

TCP连接的主动端的函数调用过程:socket()创建套接字描述符-->connect()发起连接请求-->read()/write()进行数据通信-->close()关闭套接字(断开TCP连接)。

下面我们看下断开连接的四次握手,从之前提到的函数调用过程中可以发现,无论是主动端还是被动段都可以发起断开连接,下面以主动端发起断开连接请求为例,被动端先发起是一样的。


第一次握手:主动端向被动发送fin置1,序列号为x的断开连接报文

第二次握手:被动端向主动端回溯一个ack置1,确认序列号为x+1的确认报文,主动端接到后为半关闭状态

第三次握手:被动端过一端时间后再向主动端发送fin置1,序列号为y的断开连接报文

第四次握手:主动端向被动端回溯一个ack置1,确认序列号为y+1的确认报文,此时连接为全关闭状态

整个过程是在调用close()之后由内核完成,但close()并不阻塞。这样在编程时可能会出现的这种情况,被动端在绑定网络地址时出现地址已经被占用,过一段时间后才能绑定成功,原因就是上次的四次握手还未完成。解决办法一个使用setsockopt()函数设置地址可被重复绑定,具体操作如下:

int on = 1;
setsockopt(sockfd, SOL_SOCK, SO_REUSEADDR, &on, sizeof(on));

该操作应该在socket()和bind()之间调用,其中SOL_SOCK表示是套接字的通用选项,SO_REUSEADDR表示的是地址重用选项,on=1表示开启。

APUE学习--网络编程(3)的更多相关文章

  1. Java学习——网络编程

    Java学习——网络编程 摘要:本文主要介绍了什么是网络编程,以及如何使用Java语言进行网络编程. 部分内容来自以下博客: https://www.cnblogs.com/renyuan/p/269 ...

  2. Python学习---网络编程 1217【all】

    OSI七层模型: 物理层, 数据链路层, 网络层,传输层,会话层,表达层,应用层 应用层:TFTP,HTTP,SNMP,FTP,SMTP,DNS,Telnet 等等 传输层:TCP,UDP 网络层:I ...

  3. 初识Socket通信:基于TCP和UDP协议学习网络编程

    学习笔记: 1.基于TCP协议的Socket网络编程: (1)Socket类构造方法:在客户端和服务器端建立连接 Socket s = new Socket(hostName,port);以主机名和端 ...

  4. java学习——网络编程UDP

    UDP 将数据及源和目的封装成数据包中,不需要建立连接 每个数据报的大小限制在64k内 因无连接,是不可靠协议 不需要建立连接,速度快 TCP 建立连接,形成传输数据的通道 在连接中进行大数据量传输 ...

  5. Java 网络编程学习总结

    新手一枚,Java学习中,把自己学习网络编程的知识总结一下,梳理下知识,方便日后查阅,高手莫进. 本文的主要内容: [1]    网络编程认识                [2]  TCP/IP编程 ...

  6. python网络编程学习《一》

    最近,刚实习完,很喜欢实验楼,但是自己的方向仍然不能确定,自己觉得可选择的空间很大,尽管已经是大四的人了,想到别人都在忙着买职业装,买高跟鞋面试,学习化妆什么的,看看自己,反而开始慢慢关注运动,食疗以 ...

  7. Java学习之网络编程实例

    转自:http://www.cnblogs.com/springcsc/archive/2009/12/03/1616413.html 多谢分享 网络编程 网络编程对于很多的初学者来说,都是很向往的一 ...

  8. Python学习笔记(二)网络编程的简单示例

    Python中的网络编程比C语言中要简洁很多,毕竟封装了大量的细节. 所以这里不再介绍网络编程的基本知识.而且我认为,从Python学习网络编程不是一个明智的选择.   简单的TCP连接 服务器代码如 ...

  9. 五分钟学Java:如何学习Java面试必考的网络编程

    原创声明 本文作者:黄小斜 转载请务必在文章开头注明出处和作者. 本文思维导图 简介 Java作为一门后端语言,对于网络编程的支持是必不可少的,但是,作为一个经常CRUD的Java工程师,很多时候都不 ...

随机推荐

  1. Delphi安装NT服务程序时(不出现提示信息)

    如果我们不加上"/silent",那么Delphi在安装和卸载NT服务程序时候,都会出现一个提示信息,不希望出现这个提示信息,那么使用如下命令: 1,安装:“你的nt程序 /ins ...

  2. 一个与Log4j相关的死锁(转)

    这个死锁的原因:一个动作需要两个临界对象.静态同步方法,就是一个临界对象.这种场景,静态同步方法每次只能有一个线程持有.如果存在另一个临界对象,静态同步方法中也需要获取这个临界对象.即一个动作需要两个 ...

  3. nginx的反向代理和负载均衡的区别是什么 - 开源中国社区

    nginx的反向代理和负载均衡的区别是什么 - 开源中国社区   nginx的反向代理和负载均衡的区别是什么 #user  nobody;worker_processes  1;#error_log  ...

  4. dwz框架---(2)表单回调函数

    dwz中的表单回调函数大概有下面几种: /** * 普通ajax表单提交 * @param {Object} form * @param {Object} callback * @param {Str ...

  5. TsFltMgr.sys其原因是,该系统蓝屏QQ计算机管理器!

    同事一WindowsXP系统,正常执行,关闭后,第二天无法启动.详细症状为: (1)安全模式以及带网络功能的安全模式都能够进入. (2)正常模式.还没出现WindowXP滚动栏就開始重新启动: (3) ...

  6. Linux编程实现守护进程

    Linux 守护程序 守护进程(Daemon)它是在一个特定的过程的背景进行.它独立于控制终端的和周期性地执行某些任务或待某些事件.是一种非常实用的进程. Linux的大多数server就是用守护进程 ...

  7. Eclipse shift + ctrl + F 不好用

    出现 shift + Ctrl + F  整理代码没有反应的情况,先检查下输入法是否是英文的,切换英文后再尝试.

  8. 与众不同 windows phone (15) - Media(媒体)之后台播放音频

    原文:与众不同 windows phone (15) - Media(媒体)之后台播放音频 [索引页][源码下载] 与众不同 windows phone (15) - Media(媒体)之后台播放音频 ...

  9. (一)----使用HttpClient发送HTTP请求(通过get方法获取数据)

    (一)----使用HttpClient发送HTTP请求(通过get方法获取数据) 一.HTTP协议初探: HTTP(Hypertext Transfer Protocol)中文 “超文本传输协议”,是 ...

  10. W​i​n​D​B​G​调​试​技​巧

    参考链接:http://wenku.baidu.com/view/4e58744dcf84b9d528ea7a42.html