一、服务端流程
1.创建数据报套接字[根tcp相比,第2个参数发生变化]
sockfd = socket(AF_INET,SOCK_DGRAM)
2.绑定地址
3.消息收发
data,addr = sockfd.recvfrom(buffersize) 接受udp消息。其中,data为接收到的消息内容,addr为消息发送方的地址。
n = sockfd.sendto(data,addr)发送udp消息。n为发送的字节数。
4.关闭套接字
sockfd.close()
#udp套接字服务端程序
import socket
#1.创建套接字
sockfd = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
#2.绑定
sockfd.bind(("127.0.0.1",8000))
#3.收发消息
while True:
data,addr = sockfd.recvfrom(1024)
print("Received message:%s,from %s."%(data.decode(),addr))
sockfd.sendto(b"Thanks\n",addr)
#4.关闭套接字
sockfd.close() #实际上执行不到这里
二、客户端流程
1.创建数据报套接字
sockfd = socket(AF_INET,SOCK_DGRAM)
2.收发消息:需要指定服务器的地址。
3.关闭套接字
#udp套接字客户端程序
import socket
ADDR = ("127.0.0.1",8000) #指定服务端ip和端口号
#1.创建套接字
sockfd = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
#2.收发消息
while True:
msg = input("Please input the msg:")
if not msg:
break
sockfd.sendto(msg.encode(),ADDR)
data,addr = sockfd.recvfrom(1024)
print("Received msg:%s",data.decode())
#3.关闭套接字
sockfd.close()
三、细节
UDP数据报是自带边界的,不会发生粘包
四、案例:使用udp客户端查询单词,从服务端获得单词的解释,如果没有插到该单词则得到“该单词不存在”的回复消息。 客户端可以循环查找单词,直到输入为空退出。
1.服务器端代码:
#查单词函数
def lookup_word(fd,word):
"""
从dict.txt单词表中查找单词,如果存在,则返回单词和释义,否则返回空
:param fd:文件对象
:param word: 要查找的单词(字符串)
:return:
"""
#循环读取,判断是否为要查找的单词.
for line in fd:
if word in line:
return line
else:
return
import socket
#1.创建udp套接字
sockfd = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
#2.绑定服务器ip和端口
sockfd.bind(("127.0.0.1",8000))
#3.循环接收查单词消息,发送查找结果
fd = open("dict.txt","r")
while True:
data,addr = sockfd.recvfrom(1024)
print("要查找的单词是:%s"%data.decode())
re = lookup_word(fd,data.decode())
fd.seek(0) #文件偏移量又回到开头位置
print("result:%s"%re)
if re is None:
sockfd.sendto(b"Not existed!", addr)
else:
sockfd.sendto(re.encode(), addr)
#4.关闭套接字
fd.close()
sockfd.close()
2.客户端代码
import socket
#指定服务端ip和端口
ADDR = ("127.0.0.1",8000)
#1.创建套接字
sockfd = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
#2.收发消息
while True:
word = input("请输入您要查找的单词:")
if word == "":
break
sockfd.sendto(word.encode(),ADDR)
data,addr = sockfd.recvfrom(1024)
print("查找的结果如下:%s"%data.decode())
#3.关闭套接字
sockfd.close()
四、tcp和udp套接字编程区别
1.流式套接字是以字节流的方式传输数据,数据报套接字是以数据报的形式传输;
2.tcp套接字会有粘包,udp套接字有消息边界,不会粘包。
3.tcp套接字保证了消息的完整性,而udp套接字则不会。
4.tcp套接字依赖listen、accept建立连接才能收发消息,udp套接字则不需要。
5.tcp套接字使用send、recv收发消息,udp套接字使用sendto、recvfrom收发消息。
五、套接字的一些常用属性
sockfd.type 套接字类型
sockfd.family 套接字地址类型
sockfd.getsockname()获取套接字绑定地址
sockfd.fileno()获取套接字的文件描述符
sockfd.getpeername()获取连接套接字的客户端地址(有客户端连接时才能获取,相当于addr)
sockfd.setsockopt(level,option,value):设置套接字选项。level选项类别:常用的有SOL_SOCKET等。
例:sockfd.setsockopt(SOL_SOCKET,SO_REUSEADDR,TRUE),设置套接字,当连接意外断开时,端口立即重启。
sockfd.getsockopt(level,option):获取套接字选项值
六、使用struct打包发送
1.原理:将一组简单数据进行打包,转换为bytes格式发送。或者将一组bytes格式数据,进行解析。
2.接口使用
Struct(fmt):生成结构化对象。fmt:定制的数据结构。
st.pack(v1,v2,v3):将一组数据按照指定格式打包转换成bytes(字节串).
st.unpack(bytes_data):将bytes字节串按照指定的格式解析。
也可以这么写:struck.pack(fmt,v1,v2,v3…)/struck.unpack(fmt,bytes_data)