socket又称“套接字”,应用程序通过套接字向网络发送请求或者答应网络请求,使主机或者一台计算机上的不同线程进行通信。在python中,socket跟file类似,都可以打开,读写,关闭。不同的是,file操作的是文件,socket操作的是客户端,服务器端。

在这里我们使用网络调试助手进行调试

使用“套接字”进行写数据

import socket

def main():
	# 创建套接字
	udp_s = socket.socket(family = socket.AF_INET, type = socket.SOCK_DGRAM)

	# 绑定本地发送的端口,也可以不绑定,那样的话每次发送端口就会随机
	udp_s.bind(("", 7980)) 
	
	# 发送数据,先通过 网络调试助手接收
	send_data = input("请输入要发送的数据:")
	if send_data == "q":
		break
	udp_s.sendto(send_data.encode("gbk"), ("192.168.1.1", 7979))
	
	# 关闭套接字 
	udp_s.close()

if __name__ == '__main__':
    main()
  • bind:绑定Ip地址和端口。
  • family:指定协议的地址家族,可为AF_INET或AF_UNIX。AF_INET家族包括Internet地址,AF_UNIX家族用于同一台机器上的进程间通信。
  • type:指定套接字的类型。SOCK_STREAM,TCP。SOCK_DGRAM,UDP。SOCK_RAW,原始套接字。
  • sendto:发送到数据到哪个地址,端口。
  • sendto中的发送的数据必须是字节类型的,所以这里用gbk编码,因为utf-8是window环境.

使用之前记得要点一下连接哈

python tcp socket 设置timeout后recv卡死 python socket shutdown_套接字

如果发送成功呢,这时候我们能在调试助手上看见

【Receive from 192.168.1.1 : 7980】:你好

所以我们发现,发送的数据是包含Ip地址和端口号的

使用“套接字”进行读数据

开始还是依旧的三部操作,创建套接字,读写数据,关闭套接字。

import socket

def main():
	# 创建套接字
	udp_s = socket.socket(family=socket.AF_INET, type=socket.SOCK_DGRAM)
	udp_s.bind(("", 7980))
	
	# 接收数据
	while True:
		recv_data = udp_s.recvfrom(1024) # 1024为一次接收数据的最大字节
		
		# 因为我们知道了发送/接收到的数据都是地址端口+信息
		data, info = recv_data # 拆包
		print(f"{info}:{data.decode("gbk")}") # 解码为汉字
	udp_s.close()
if __name__ == '__main__':
    main()

开启程序之后,直接在 调试工具里面发送数据,程序就能接收到了

调试工具的目标端口,就是接收方的绑定端口

python tcp socket 设置timeout后recv卡死 python socket shutdown_套接字_02

聊天器效果

开启后,在调试工具中进行调试,可以实现聊天效果。

import socket

def send_data(udp_s):
    send_d = input("请输入数据:")
    udp_s.sendto(send_d.encode("gbk"), ("192.168.1.1", 7979))


def recv_data(udp_s):
    recv_d = udp_s.recvfrom(1024)
    print(recv_d)
    # data, info = recv_data
    # print(f"{info}:{data.decode('gbk')}")
def main():
    udp_s = socket.socket(family=socket.AF_INET, type=socket.SOCK_DGRAM)

    # 绑定本地ip与端口
    udp_s.bind(("", 7980))
    # 3.先发送再接收
    while True:

        send_data(udp_s)

        recv_data(udp_s)

    udp_s.close()

if __name__ == '__main__':
    main()