一文读懂Socket通信原理

Posted Python那些事

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了一文读懂Socket通信原理相关的知识,希望对你有一定的参考价值。


来源:wzhvictor   链接:

https://segmentfault.com/a/1190000013712747


什么是Socket?

Socket的中文翻译过来就是“套接字”。套接字是什么,我们先来看看它的英文含义:插座。

实际上,Socket是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口,供应用层调用实现进程在网络中的通信。Socket起源于UNIX,在Unix一切皆文件的思想下,进程间通信就被冠名为文件描述符(file desciptor),Socket是一种“打开—读/写—关闭”模式的实现,服务器和客户端各自维护一个“文件”,在建立连接打开后,可以向文件写入内容供对方读取或者读取对方内容,通讯结束时关闭文件。

另外我们经常说到的Socket所在位置如下图:

Socket通信过程

Socket保证了不同计算机之间的通信,也就是网络通信。对于网站,通信模型是服务器与客户端之间的通信。两端都建立了一个Socket对象,然后通过Socket对象对数据进行传输。通常服务器处于一个无限循环,等待客户端的连接。

一图胜千言,下面是面向连接的TCP时序图:

一文读懂Socket通信原理

客户端过程:

客户端的过程比较简单,创建Socket,连接服务器,将Socket与远程主机连接(注意:只有TCP才有“连接”的概念,一些Socket比如UDP、ICMP和ARP没有“连接”的概念),发送数据,读取响应数据,直到数据交换完毕,关闭连接,结束TCP对话。

import socket
import sys

if __name__ == '__main__':
   sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  # 创建Socket连接
   sock.connect(('127.0.0.1', 8001))  # 连接服务器
   while True:
       data = input('Please input data:')
       if not data:
           break
       try:
           sock.sendall(data)
       except socket.error as e:
           print('Send Failed...', e)
           sys.exit(0)
       print('Send Successfully')

       res = sock.recv(4096)  # 获取服务器返回的数据,还可以用recvfrom()、recv_into()等
       print(res)
   sock.close()
sock.sendall(data)

这里也可用send()方法:不同在于sendall()在返回前会尝试发送所有数据,并且成功时返回None,而send()则返回发送的字节数量,失败时都抛出异常。

服务端过程:

import socket
import sys

if __name__ == '__main__':
   sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  # 创建Socket连接(TCP)
   print('Socket Created')

   try:
       sock.bind(('127.0.0.1', 8001))  # 配置Socket,绑定IP地址和端口号
   except socket.error as e:
       print('Bind Failed...', e)
       sys.exit(0)

   sock.listen(5)  # 设置最大允许连接数,各连接和Server的通信遵循FIFO原则

   while True:  # 循环轮询Socket状态,等待访问
       conn, addr = sock.accept()
       try:
           conn.settimeout(10)  # 获得一个连接,然后开始循环处理这个连接发送的信息

           # 如果要同时处理多个连接,则下面的语句块应该用多线程来处理
           while True:
               data = conn.recv(1024)
               print('Get value ' + data, end='\n\n')
               if not data:
                   print('Exit Server', end='\n\n')
                   break
               conn.sendall('OK')  # 返回数据
       except socket.timeout:  # 建立连接后,该连接在设定的时间内没有数据发来,就会引发超时
           print('Time out')

       conn.close()  # 当一个连接监听循环退出后,连接可以关掉
   sock.close()
conn, addr = sock.accept()
data = conn.recv(1024)

接下来是处理阶段,服务器和客户端通过send()recv()通信(传输数据)。
服务器调用
send(),并采用字符串形式向客户发送信息,send()返回已发送的字符个数。
服务器调用
recv()从客户接收信息。调用recv()时,服务器必须指定一个整数,它对应于可通过本次方法调用来接收的最大数据量。recv()在接收数据时会进入“blocked”状态,最后返回一个字符串,用它表示收到的数据。如果发送的数据量超过了recv()所允许的,数据会被截短。多余的数据将缓冲于接收端,以后调用recv()时,多余的数据会从缓冲区删除(以及自上次调用recv()以来,客户可能发送的其它任何数据)。传输结束,服务器调用Socket的close()关闭连接。

TCP三次握手的Socket过程:

一文读懂Socket通信原理

  • 服务器调用socket()bind()listen()完成初始化后,调用accept()阻塞等待;

  • 客户端Socket对象调用connect()向服务器发送了一个SYN并阻塞;

  • 服务器完成了第一次握手,即发送SYN和ACK应答;

  • 客户端收到服务端发送的应答之后,从connect()返回,再发送一个ACK给服务器;

  • 服务器Socket对象接收客户端第三次握手ACK确认,此时服务端从accept()返回,建立连接。


接下来就是两个端的连接对象互相收发数据。

TCP四次挥手的Socket过程:


  • 某个应用进程调用close()主动关闭,发送一个FIN;

  • 另一端接收到FIN后被动执行关闭,并发送ACK确认;

  • 之后被动执行关闭的应用进程调用close()关闭Socket,并也发送一个FIN;

  • 接收到这个FIN的一端向另一端ACK确认。


上面的代码是简单的演示Socket的基本函数使用,其实不管有多复杂的网络程序,这些基本函数都会用到。上面的服务端代码只有处理完一个客户端请求才会去处理下一个客户端的请求,这样的服务器处理能力很弱,而实际中服务器都需要有并发处理能力,为了达到并发处理,服务器就需要fork一个新的进程或者线程去处理请求。


(完)


看完本文有收获?请转发分享给更多人

关注「Python那些事」,做全栈开发工程师

以上是关于一文读懂Socket通信原理的主要内容,如果未能解决你的问题,请参考以下文章

深入浅出——零基础一文读懂DeepSORT(原理篇)

一文读懂Zookeeper原子广播原理 深入浅出Zookeeper

一文读懂Linux任务间调度原理和整个执行过程

一文读懂keepalive的工作原理

一文教你读懂光纤跳线和双绞线

一文读懂遗传算法工作原理(附Python实现)