网络上的两个程序通过一个双向的通信连接实现数据的交换,连接的一端称为一个Socket。Socket的本质是编程接口(API)。Socket通常也称为“套接字”,由IP地址和端口组成,是一个通信链的句柄,应用程序通过“套接字”向网络中发出请求或者应答网络请求。例如,在浏览器地址栏中输入“http://www.baidu.com/”时,会打开一个套接字,然后连接到“http://www.baidu.com/”并读取响应的页面然后显示出来。
每一个Socket都有一个本地唯一的Socket号,由操作系统分配。“套接字”有3中类型:流式套接字(SOCK_STREAM)、数据报套接字(SOCK_DGRAM)和原始套接字。
流式套接字提供可靠的、面向连接的通信流,如同TCP协议。
数据报套接字定义了一种无连接的服务,数据通过相互独立的报文进行传输,是无序的,并且不保证可靠性。数据报套接字使用用户数据报协议UDP,数据只是简单地传送到对方。
原始套接字允许对低层协议如IP或ICMP直接访问,功能强大,但是没有流式套接字和数据报套接字使用方便,一般的程序不涉及原始套接字。
TCP连接
'''
TCP客户端步骤
创建客户端套接字对象 socket.socket()
和服务器端套接字进行连接 connect()
发送数据 send()
接收数据 recv()
关闭客户端套接字 close()
创建客户端套接字
socket.socket(AddressFamily, Type)
AddressFamily 表示IP地址类型,分为IPv4和IPv6
Type 表示传输协议类型
socket.socket(socket.AF_INET, socket.SOCK_STREAM)
AF_INET:表示IPv4 AF_INET6:表示IPv6
SOCK_STREAM:表示TCP传输协议
SOCK_DGRAM:表示UDP传输协议
连接服务端套接字
connect((host, port)) 表示和服务端套接字建立连接, host是服务器ip地址,port是应用程序的端口号
收发数据
send(data) 表示发送数据,data是二进制数据
recv(buffersize) 表示接收数据, buffersize是每次接收数据的长度
'''
import socket
# 创建套接字
tcp_client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 连接服务端套接字
tcp_client_socket.connect(('192.168.126.100', 6666))
# 准备数据
data = 'hello world'.encode('utf-8')
# 发送数据
tcp_client_socket.send(data)
# 接收数据
# windows中是gbk,ubuntu是utf-8
data = tcp_client_socket.recv(1024)
print(data.decode('utf-8'))
# 关闭连接
tcp_client_socket.close()
'''
TCP服务端步骤
创建服务端套接字
绑定端口号
设置监听
等待客户端连接请求
接收数据
发送数据
关闭套接字
创建套接字:和客户端相同
socket.socket(AddressFamily, Type)
绑定端口
bind((host, port)) ip一般为空不指定,表示本机的任何一个ip地址都可以
监听
listen(basklog) backlog参数表示最大等待建立连接的个数,最大有多少个客户在等待
等待客户端请求
accept() 表示等待接受客户端的连接请求,相当于执行三次握手
发送数据
send(data)
接受数据
recv(buffersize) buffersize 是每次接收数据的长度
'''
import socket
# 创建套接字
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 设置端口号复用,在tcp四次挥手之后端口不会立即释放,这条命令就是让程序退出后,端口立即释放
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 绑定端口
# 如果不指定ip,默认绑定的是本地ip地址
tcp_server_socket.bind(('', 5555))
# 监听
# 128:最大等待建立连接的个数, 提示: 目前是单任务的服务端,同一时刻只能服务与一个客户端,后续
# 使用多任务能够让服务端同时服务于多个客户端
# 不需要让客户端进行等待建立连接
# listen后的这个套接字只负责接收客户端连接请求,不能收发消息,收发消息使用返回的这个新套接字来完成
tcp_server_socket.listen(128)
# 等待客户建立连接的请求,此时执行三次握手
# 等待客户端建立连接的请求, 只有客户端和服务端建立连接成功代码才会解阻塞,代码才能继续往下执行
# 1. 专门和客户端通信的套接字: service_client_socket
# 2. 客户端的ip地址和端口号: ip_port
service_client_socket, ip_port = tcp_server_socket.accept()
print(f'客户端ip:{ip_port[0]},客户端端口:{ip_port[1]}')
# 接收数据
recv_data = service_client_socket.recv(1024)
recv_content = recv_data.decode('utf-8')
print('接收的数据:', recv_content, len(recv_content))
# 发送数据
send_data = '这是服务端发送的数据......'
encode_data = send_data.encode('utf-8')
service_client_socket.send(encode_data)
# 关闭服务与客户端的套接字, 终止和客户端通信的服务,相当于关闭客户端套接字
service_client_socket.close()
# 关闭服务端的套接字, 终止和客户端提供建立连接请求的服务
tcp_server_socket.close()
UDP连接
'''
udp客户端实现
'''
import socket
c = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
while True:
try:
msg = input('>>>')
if msg == 0:
continue
c.sendto(msg.encode('utf-8'), ('127.0.0.1', 5555)) # UDP不需要构成连接,直接发送即可
data, s_addr = c.recvfrom(1024) # c_addr是一个地址,发送消息的客户端的ip和端口构成的元组
print(data.decode('utf-8'))
except KeyboardInterrupt:
break
c.close()
'''
udp服务器实现
'''
import socket
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.bind(("", 5555))
print("Server Opening....")
while True:
try:
data, c_addr = s.recvfrom(1024)
print('from:', c_addr)
print(data.decode('utf-8'))
send_data = 'hello, this is server'
s.sendto(send_data.encode('utf-8'), c_addr) # 发送信息,一个参数是信息,一个是目标地址和端口构成的元组
except KeyboardInterrupt:
break
print('Server Closing')
s.close()