socket
(1)scoket层在哪

(2)什么是socket
- Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。
- 在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面
- 对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。
- 所以,我们无需深入理解tcp/udp协议,socket已经为我们封装好了,我们只需要遵循socket的规定去编程,写出的程序自然就是遵循tcp/udp标准的。
- 也有人将socket说成ip+port
- ip是用来标识互联网中的一台主机的位置
- 而port是用来标识这台机器上的一个应用程序
- ip地址是配置到网卡上的
- 而port是应用程序开启的
- ip与port的绑定就标识了互联网中独一无二的一个应用程序
- 而程序的pid是同一台机器上不同进程或者线程的标识
(3)套接字发展史及分类
- 套接字起源于 20 世纪 70 年代加利福尼亚大学伯克利分校版本的 Unix,即人们所说的 BSD Unix。
- 因此,有时人们也把套接字称为“伯克利套接字”或“BSD 套接字”。
- 一开始,套接字被设计用在同一台主机上多个应用程序之间的通讯。
- 这也被称进程间通讯,或IPC。套接字有两种(或者称为有两个种族)分别是
(1)基于文件类型的套接字家族
- 套接字家族的名字:
- unix一切皆文件
- 基于文件的套接字调用的就是底层的文件系统来取数据
- 两个套接字进程运行在同一机器
- 可以通过访问同一个文件系统间接完成通信
(2)基于网络类型的套接字家族
- 套接字家族的名字:
- 还有AF_INET6被用于ipv6,还有一些其他的地址家族,不过,他们要么是只用于某个平台,要么就是已经被废弃,或者是很少被使用,或者是根本没有实现
- 所有地址家族中,AF_INET是使用最广泛的一个
- python支持很多种地址家族,但是由于我们只关心网络编程,所以大部分时候我么只使用AF_INET
套接字工作流程
- 一个生活中的场景。
- 你要打电话给一个朋友,先拨号,朋友听到电话铃声后提起电话,这时你和你的朋友就建立起了连接,就可以讲话了。
- 等交流结束,挂断电话结束此次交谈。 生活中的场景就解释了这工作原理。

(1)服务端流程
- 先从服务器端说起。
- 服务器端先初始化Socket
- 然后与端口绑定(bind),对端口进行监听(listen)
- 调用accept阻塞,等待客户端连接。
- 在这时如果有个客户端初始化一个Socket
- 然后连接服务器(connect)
- 如果连接成功,这时客户端与服务器端的连接就建立了。
- 客户端发送数据请求,服务器端接收请求并处理请求
- 然后把回应数据发送给客户端,客户端读取数据
- 最后关闭连接,一次交互结束
- socket()模块函数用法
| import socket |
| socket.socket(socket_family,socket_type,protocal=0) |
| |
| |
| |
| tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) |
| |
| |
| udpSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) |
| |
| |
| |
(2)服务端套接字函数
- s.bind() 绑定(主机,端口号)到套接字
- s.listen() 开始TCP监听
- s.accept() 被动接受TCP客户的连接,(阻塞式)等待连接的到来
(3)客户端套接字函数
- s.connect() 主动初始化TCP服务器连接
- s.connect_ex() connect()函数的扩展版本,出错时返回出错码,而不是抛出异常
(4)公共用途的套接字函数
- s.recv() 接收TCP数据
- s.send() 发送TCP数据(send在待发送数据量大于己端缓存区剩余空间时,数据丢失,不会发完)
- s.sendall() 发送完整的TCP数据(本质就是循环调用send,sendall在待发送数据量大于己端缓存区剩余空间时,数据不丢失,循环调用send直到发完)
- s.recvfrom() 接收UDP数据
- s.sendto() 发送UDP数据
- s.getpeername() 连接到当前套接字的远端的地址
- s.getsockname() 当前套接字的地址
- s.getsockopt() 返回指定套接字的参数
- s.setsockopt() 设置指定套接字的参数
- s.close() 关闭套接字
(5)面向锁的套接字方法
- s.setblocking() 设置套接字的阻塞与非阻塞模式
- s.settimeout() 设置阻塞套接字操作的超时时间
- s.gettimeout() 得到阻塞套接字操作的超时时间
(6)面向文件的套接字的函数
- s.fileno() 套接字的文件描述符
- s.makefile() 创建一个与该套接字相关的文件
基于TCP的套接字
(1)方法简介
| server = socket() |
| server.bind() |
| server.listen() |
| inf_loop: |
| conn = server.accept() |
| comm_loop: |
| conn.recv()/conn.send() |
| conn.close() |
| server.close() |
| client = socket() |
| client.connect() |
| comm_loop: |
| client.send()/client.recv() |
| client.close() |
(2)打电话模型
(1)服务端
| import socket |
| |
| |
| server = socket.socket() |
| |
| |
| IP = '127.0.0.1' |
| PORT = 8888 |
| |
| server.bind((IP, PORT)) |
| |
| |
| |
| server.listen(5) |
| |
| |
| conn, addr = server.accept() |
| |
| |
| |
| |
| |
| |
| from_to_client_msg = conn.recv(1024) |
| from_to_client_msg = from_to_client_msg.decode('utf-8') |
| print(f'from_to_client_msg:>>> {from_to_client_msg}') |
| |
| |
| send_to_client_msg = f'我是服务端' |
| send_to_client_msg = send_to_client_msg.encode('utf-8') |
| conn.send(send_to_client_msg) |
| |
| |
| conn.close() |
| |
| |
| server.close() |
(2)客户端
| import socket |
| |
| |
| client = socket.socket() |
| |
| |
| IP = '127.0.0.1' |
| PORT = 8888 |
| client.connect((IP, PORT)) |
| |
| |
| send_to_server_msg = f'我是客户端' |
| send_to_server_msg = send_to_server_msg.encode('utf-8') |
| client.send(send_to_server_msg) |
| |
| |
| from_server_msg = client.recv(1024) |
| |
| from_server_msg = from_server_msg.decode('utf-8') |
| print(f'from_server_msg: >>>{from_server_msg}') |
| |
| |
| client.close() |
单次模版总结
(1)服务端模版
| from socket import socket |
| |
| |
| server = socket() |
| |
| |
| IP = '127.0.0.1' |
| PORT = 8080 |
| server.bind((IP, PORT)) |
| |
| |
| server.listen(5) |
| |
| |
| conn, addr = server.accept() |
| |
| |
| from_client_msg = conn.recv(1024) |
| from_client_msg = from_client_msg.decode('utf-8') |
| print(f'来自客户端的信息 :>>> {from_client_msg}') |
| |
| |
| send_to_client_msg = '我是服务端' |
| send_to_client_msg = send_to_client_msg.encode('utf-8') |
| conn.send(send_to_client_msg) |
| |
| |
| conn.close() |
| |
| |
| server.close() |
(2)客户端模版
| from socket import socket |
| |
| |
| client = socket() |
| |
| |
| IP = '127.0.0.1' |
| PORT = 9080 |
| client.connect((IP, PORT)) |
| |
| |
| send_to_server_msg = f'我是客户端' |
| send_to_server_msg = send_to_server_msg.encode('utf-8') |
| client.send(send_to_server_msg) |
| |
| |
| from_server_msg = client.recv(1024) |
| |
| from_server_msg = from_server_msg.decode('utf-8') |
| print(f'来自服务端的信息 : >>>{from_server_msg}') |
(3)打电话模型持续交互
(1)服务端
| import socket |
| |
| |
| server = socket.socket() |
| |
| |
| IP = '127.0.0.1' |
| PORT = 8888 |
| |
| server.bind((IP, PORT)) |
| |
| |
| server.listen(5) |
| |
| while True: |
| |
| conn, addr = server.accept() |
| |
| |
| |
| from_to_client_msg = conn.recv(1024) |
| from_to_client_msg = from_to_client_msg.decode('utf-8') |
| print(f'来自客户端:>>> {from_to_client_msg}') |
| |
| |
| |
| while True: |
| send_to_client_msg = input(f'请输入发送的消息 :>>>').strip() |
| if not send_to_client_msg: |
| print(f'当前不能为空!请重新输入!') |
| continue |
| send_to_client_msg = send_to_client_msg.encode('utf-8') |
| conn.send(send_to_client_msg) |
| break |
| |
| |
| conn.close() |
| |
| server.close() |
(2)客户端
| import socket |
| |
| |
| while True: |
| |
| client = socket.socket() |
| |
| |
| IP = '127.0.0.1' |
| PORT = 8888 |
| client.connect((IP, PORT)) |
| |
| |
| send_to_server_msg = input(f'请输入发送的消息 :>>>').strip() |
| if not send_to_server_msg: |
| print(f'当前不能为空!请重新输入!') |
| continue |
| send_to_server_msg = send_to_server_msg.encode('utf-8') |
| client.send(send_to_server_msg) |
| |
| |
| from_server_msg = client.recv(1024) |
| |
| from_server_msg = from_server_msg.decode('utf-8') |
| print(f'来自服务端: >>>{from_server_msg}') |
| |
| |
| client.close() |
循环模版总结
(1)服务端模版
| from socket import socket |
| |
| |
| server = socket() |
| |
| |
| server.bind(('127.0.0.1', 8888)) |
| |
| |
| server.listen(5) |
| |
| while True: |
| |
| |
| |
| |
| conn, addr = server.accept() |
| |
| |
| msg = conn.recv(1024) |
| print(f'来自客户端:>>> {msg.decode("utf-8")}') |
| |
| |
| while True: |
| send_to_client_msg = input(f'请输入发送的消息 :>>>').strip() |
| if not send_to_client_msg: |
| print(f'当前内容为空!请重新输入!') |
| continue |
| conn.send(send_to_client_msg.encode('utf-8')) |
| break |
| |
| |
| conn.close() |
| |
| |
| server.close() |
(2)客户端模版
| from socket import socket |
| |
| while True: |
| |
| client = socket() |
| |
| |
| client.connect(('127.0.0.1', 8888)) |
| |
| |
| send_to_server_msg = input(f'请输入发送的消息 :>>>').strip() |
| if not send_to_server_msg: |
| print(f'当前不能为空!请重新输入!') |
| continue |
| client.send(send_to_server_msg.encode('utf-8')) |
| |
| |
| from_server_msg = client.recv(1024) |
| |
| |
| from_server_msg = from_server_msg.decode('utf-8') |
| print(f'来自服务端: >>>{from_server_msg}') |
| |
| |
| client.close() |
基于UDP的套接字
(1)循环模版
(1)服务端模版
| from socket import socket, AF_INET, SOCK_DGRAM |
| |
| |
| server = socket(AF_INET, SOCK_DGRAM) |
| |
| |
| server_address = ('127.0.0.1', 8888) |
| server.bind(server_address) |
| |
| while True: |
| |
| data, client_address = server.recvfrom(1024) |
| |
| |
| decoded_data = data.decode('utf-8') |
| print(f'来自客户端的信息:>>> {decoded_data}') |
| |
| |
| while True: |
| response_msg = input(f'请输入发送的消息 :>>>').strip() |
| if not response_msg: |
| print(f'当前不能为空!请重新输入!') |
| continue |
| server.sendto(response_msg.encode('utf-8'), client_address) |
| break |
| |
| |
| server.close() |
(2)客户端模版
| from socket import socket, AF_INET, SOCK_DGRAM |
| |
| |
| client = socket(AF_INET, SOCK_DGRAM) |
| |
| |
| server_address = ('127.0.0.1', 8888) |
| |
| while True: |
| |
| send_to_server_msg = input(f'请输入发送的消息 :>>>').strip() |
| if not send_to_server_msg: |
| print(f'当前不能为空!请重新输入!') |
| continue |
| |
| client.sendto(send_to_server_msg.encode('utf-8'), server_address) |
| |
| |
| data, server_response_address = client.recvfrom(1024) |
| decoded_data = data.decode('utf-8') |
| print(f'来自服务器的信息:>>> {decoded_data}') |
| |
| |
| client.close() |
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通