• UDP网络通信(发送端+接收端)实例 —— Python


    简介
    在网络通信编程中,用的最多的就是UDP和TCP通信了,原理这里就不分析了,网上介绍也很多,这里简单列举一下各自的优缺点和使用场景

    通信方式优点缺点适用场景
    UDP及时性好,快速视网络情况,存在丢包

    嵌入式设备通信,实时控制

    场景

    TCP丢包会自动重发,理论上不用担心丢包问题延时相对大一些

    通信可靠性场景,比如IoT设备

    控制,状态同步

    一、socket
    我们要进行网络通信,那么就要用到socket,socket即网络套接字,应用程序可以通过它发送或接收数据,可对其进行像对文件一样的打开、读写和关闭等操作。
    在 Python 中,使用socket 模块的函数 socket 就可以创建一个socket对象,socket()函数的参数分别有family, type, proto。

    1.其中family参数是指协议域,又称为协议族(family),常用的协议族有,AF_INET、AF_INET6、...等等,AF_INET指ipv4,AF_INET6即为ipv6;
    2.然后是type,type指定socket类型,有SOCK_STREAM(流式套接字,主要用于 TCP 协议)和SOCK_DGRAM(数据报套接字,主要用于 UDP 协议)等等;
    3.proto就是指定的协议,常用的协议有,IPPROTO_TCP、IPPTOTO_UDP、IPPROTO_SCTP、IPPROTO_TIPC等,它们分别对应TCP传输协议、UDP传输协议、STCP传输协议、TIPC传输协议,但是type和proto不可以随意组合,当proto参数为0或者不填时,会自动选择type类型对应的默认协议。

    二、UDP发送数据
    首先我们要导入socket包

    import socket

    创建一个udp套接字,ipv4协议,使用SOCK_DGRAM参数,不填proto,就会默认自动选择udp协议;

    1. # 1、创建一个UDP套接字
    2. udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

    然后我们把要接收数据的那一端的ip地址和端口号放在一个元组里准备好

    1. ​​​# 2. 准备接收方的地址和端口,'127.0.0.1:9999'表示目的ip地址,9999表示目的端口号
    2. udp_addr = ('127.0.0.1', 9999)  # 注意这是一个元组,其中ip地址是字符串,端口号是数字​

      准备好后就可以使用sendto函数进行发送了,要注意,需要对字符串进行编码才可以发送

    1. # 3. 发送数据到指定的ip和端口
    2. udp_socket.sendto("Hello,I am a UDP socket.".encode('utf-8'), dest_addr)

    发送完就可以关闭套接字了

    1. # 4. 关闭套接字
    2. udp_socket.close()

    例程一:UDP server端,UDP数据接收

    1. #!/usr/bin/python3
    2. # -*- coding: utf-8 -*-
    3. """
    4. udp通信例程:udp server端,修改udp_addr元组里面的ip地址,即可实现与目标机器的通信,
    5. 此处以单机通信示例,ip为127.0.0.1,实际多机通信,此处应设置为目标客户端ip地址
    6. """
    7. from time import sleep
    8. import socket
    9. def main():
    10.     # udp 通信地址,IP+端口号
    11.     udp_addr = ('127.0.0.1', 9999)
    12.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    13.     # 绑定端口
    14.     udp_socket.bind(udp_addr)
    15.     # 等待接收对方发送的数据
    16.     while True:
    17.         recv_data = udp_socket.recvfrom(1024)  # 1024表示本次接收的最大字节数
    18.         # 打印接收到的数据
    19.         print("[From %s:%d]:%s" % (recv_data[1][0], recv_data[1][1], recv_data[0].decode("utf-8")))
    20. if __name__ == '__main__':
    21.     print("当前版本: ", __version__)
    22.     print("udp server ")
    23.     main()

     代码解析 

    1.socket函数中第二个参数就是通信类型,此处SOCK_DGRAM 就是指定使用UDP通信
    2.服务端需要使用bind函数绑定端口,客户端不需要,因为客户端发送的时候已经带了端口参数

    例程二:UDP client端,UDP数据发送

    1. #!/usr/bin/python3
    2. # -*- coding: utf-8 -*-
    3. """
    4. udp通信例程:udp client端,修改udp_addr元组里面的ip地址,即可实现与目标机器的通信,
    5. 此处以单机通信示例,ip为127.0.0.1,实际多机通信,此处应设置为目标服务端ip地址
    6. """
    7. from time import sleep
    8. import socket
    9. def main():
    10.     # udp 通信地址,IP+端口号
    11.     udp_addr = ('127.0.0.1', 9999)
    12.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    13.     # 发送数据到指定的ip和端口,每隔1s发送一次,发送10次
    14.     for i in range(10):
    15.         udp_socket.sendto(("Hello,I am a UDP socket for: " + str(i)) .encode('utf-8'), udp_addr)
    16.         print("send %d message" % i)
    17.         sleep(1)
    18.     # 5. 关闭套接字
    19.     udp_socket.close()
    20. if __name__ == '__main__':
    21.     print("当前版本: ", __version__)
    22.     print("udp client ")
    23.     main()


    例程三:多线程实现UDP数据收发

    1. #!/usr/bin/python3
    2. # -*- coding: utf-8 -*-
    3. """
    4. python多线程通信
    5. """
    6. from time import sleep
    7. import socket
    8. import threading
    9. # 定义全局变量
    10. t1_count = 0
    11. t2_count = 0
    12. def udp_received_hundle(s):
    13.     global t1_count
    14.     print("this is thread 1 running")
    15.     while True:
    16.         t1_count += 1
    17.         print("thread 1 第 %s 次运行" % t1_count)
    18.         recv_data = s.recvfrom(1024)  # 1024表示本次接收的最大字节数
    19.         # 打印接收到的数据
    20.         print("[From %s:%d]:%s" % (recv_data[1][0], recv_data[1][1], recv_data[0].decode("utf-8")))
    21. def udp_send_hundle(s):
    22.     global t2_count
    23.     print("this is thread 2 running")
    24.     while True:
    25.         t2_count += 1
    26.         print("")
    27.         print("thread 2 第 %s 次运行" % t2_count)
    28.         s.sendto(("Hello,I am a UDP socket for: " + str(t2_count)).encode('utf-8'), udp_addr)
    29.         print("send %d message" % t2_count)
    30.         print("")
    31.         sleep(1)
    32. if __name__ == '__main__':
    33.     print("当前版本: ", __version__)
    34.     # 初始化
    35.     # udp 通信地址,IP+端口号
    36.     udp_addr = ('127.0.0.1', 9999)
    37.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    38.     # 绑定端口:
    39.     udp_socket.bind(udp_addr)
    40.     # 定义线程
    41.     thread_list = []
    42.     t1 = threading.Thread(target=udp_received_hundle, args=(udp_socket, ))
    43.     thread_list.append(t1)
    44.     t2 = threading.Thread(target=udp_send_hundle, args=(udp_socket, ))
    45.     thread_list.append(t2)
    46.     for t in thread_list:
    47.         t.setDaemon(True)
    48.         t.start()
    49.     for t in thread_list:
    50.         t.join()
    51.     print("exit all task.")
    52.     print('all process end.')

    代码解析
    这里用到了多线程,虽然python中的多线程是假的多线程,实际上是一个线程分时复用,这里我们不深究,如果平常用到也就几个小任务跑一跑,抄我这个作业就ok。
    多线程实际上是从t.join()后才开始正式运行的,这里一定要注意,不能漏了这个函数。
    udp的收发与上面的例程几乎是一样的。
    代码运行效果如下

    1. 当前版本:  1.0.0
    2. this is thread 1 running
    3. thread 11 次运行
    4. this is thread 2 running
    5. thread 21 次运行
    6. send 1 message
    7. [From 127.0.0.1:9999]:Hello,I am a UDP socket for: 1
    8. thread 12 次运行
    9. thread 22 次运行
    10. send 2 message
    11. [From 127.0.0.1:9999]:Hello,I am a UDP socket for: 2
    12. thread 13 次运行
    13. thread 23 次运行
    14. send 3 message
    15. [From 127.0.0.1:9999]:Hello,I am a UDP socket for: 3
    16. thread 14 次运行
    17. thread 24 次运行
    18. send 4 message
    19. [From 127.0.0.1:9999]:Hello,I am a UDP socket for: 4
    20. thread 15 次运行
    21. thread 25 次运行
    22. send 5 message

     使用网络调试助手,测试程序
    注意,如果不是在本机windows系统上运行python程序,在Ubuntu虚拟机或者其他局域网内的机器上运行,要把windows的防火墙关了!!!
    然后我们让其每隔一秒发送一次,发送10次,发送成功

    完整代码:

    1. #!/usr/bin/env python
    2. # -*- coding:utf-8 -*-
    3. # Author: William
    4. import socket,time
    5. def main():
    6.     # 1、创建一个UDP套接字
    7.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    8.     # 2. 准备接收方的地址和端口,'192.168.0.107'表示目的ip地址,8080表示目的端口号
    9.     dest_addr = ('192.168.8.226', 12341)  # 注意这是一个元组,其中ip地址是字符串,端口号是数字
    10.     # 3. 发送数据到指定的ip和端口
    11.     for i in range(10):
    12.         udp_socket.sendto("Hello,I am a UDP socket.".encode('utf-8'), dest_addr)
    13.         time.sleep(1)
    14.     # 4. 关闭套接字
    15.     udp_socket.close()
    16. if __name__ == '__main__':
    17.     main()

    三、UDP接收数据
    在之前发送数据的时候,我们可以看到,其端口号是一直在变得,那么我们要接收数据,就需要知道其端口号是什么,所以我们要先固定一个端口号,使用bind函数

    1. # 2. 绑定本地的相关信息,如果不绑定,则系统会随机分配一个端口号
    2. local_addr = ('', 12344)  # ip地址和端口号,ip一般不用写,表示本机的任何一个ip
    3. udp_socket.bind(local_addr)

    接收数据使用recvfrom函数,其参数为接收的最大数据长度

    1. # 3. 等待接收对方发送的数据
    2. recv_data = udp_socket.recvfrom(1024)  # 1024表示本次接收的最大字节数

    接收完后将其打印出来:

    1. # 4、打印接收到的数据
    2. print(recv_data)

    运行,通过网络调试助手发送数据

     可以看到,打印出来的信息是一个元组,第一项接收到的字符串,第二项也是一个元组,包含对方的IP地址和端口号
    完整代码:

    1. #!/usr/bin/env python
    2. # -*- coding:utf-8 -*-
    3. # Author: William
    4. import socket,time
    5. def main():
    6.     # 1、创建一个UDP套接字
    7.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    8.     # 2. 绑定本地的相关信息,如果不绑定,则系统会随机分配一个端口号
    9.     local_addr = ('', 12344)  # ip地址和端口号,ip一般不用写,表示本机的任何一个ip
    10.     udp_socket.bind(local_addr)
    11.     # 3. 等待接收对方发送的数据
    12.     recv_data = udp_socket.recvfrom(1024)  # 1024表示本次接收的最大字节数
    13.     # 4、打印接收到的数据
    14.     print(recv_data)
    15.     # 5. 关闭套接字
    16.     udp_socket.close()
    17. if __name__ == '__main__':
    18.     main()


    四、UDP收发数据
    实现这样一个功能,通过UDP发送10次消息,然后等待接收,将接收的数据及其来源打印出来:

    完成代码:

    1. #!/usr/bin/env python
    2. # -*- coding:utf-8 -*-
    3. # Author: William
    4. import socket,time
    5. def main():
    6.     # 1、创建一个UDP套接字
    7.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    8.     # 2. 绑定本地的相关信息,如果不绑定,则系统会随机分配一个端口号
    9.     udp_socket.bind(('', 12344))
    10.     # 3. 发送数据到指定的ip和端口,每隔1s发送一次,发送10次
    11.     for i in range(10):
    12.         udp_socket.sendto("Hello,I am a UDP socket.".encode('utf-8'), ('192.168.8.226', 12341))
    13.         time.sleep(1)
    14.     # 4. 等待接收对方发送的数据
    15.     while(True):
    16.         recv_data = udp_socket.recvfrom(1024)
    17.         # 打印接收到的数据
    18.         print("[From %s:%d]:%s"%(recv_data[1][0],recv_data[1][1],recv_data[0].decode("utf-8")))
    19.     # 5. 关闭套接字
    20.     udp_socket.close()
    21. if __name__ == '__main__':
    22.     main()


    五、同时收发数据
    现在实现这样一个功能,即运行程序,然后在控制台输入字符串发送出去,同时,还可以接收数据,我使用多线程来实现这个程序,不过要实现方便接收,我们在程序的开始,将IP地址和端口号打印出来,实现效果如下:

    实现代码:

    1. #!/usr/bin/env python
    2. # -*- coding:utf-8 -*-
    3. # Author: William
    4. import socket,time,threading
    5. def recv_thread(socket):
    6.     # 等待接收对方发送的数据
    7.     while(True):
    8.         try:
    9.             recv_data = socket.recvfrom(1024)
    10.             # 打印接收到的数据
    11.             print("[From %s:%d]:%s"%(recv_data[1][0],recv_data[1][1],recv_data[0].decode("utf-8")))
    12.         except Exception:
    13.             break
    14. def main():
    15.     # 1、创建一个UDP套接字
    16.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    17.     # 2、绑定本地的相关信息,如果不绑定,则系统会随机分配一个端口号
    18.     udp_socket.bind(('', 12344))
    19.     # 3、打印本机ip地址和端口号
    20.     print("local ipaddr and port->",socket.gethostbyname(socket.gethostname())+":12344")
    21.     # 4、创建一个线程,用来接收数据
    22.     t = threading.Thread(target=recv_thread, args=(udp_socket,))
    23.     t.start()
    24.     # 5、等待输入数据,然后发送出去,直到输入的数据为'quit'
    25.     while(True):
    26.         print("please input a string.input 'quit' to quit.")
    27.         send_data = input()
    28.         if send_data == "quit":
    29.             break
    30.         else:
    31.             udp_socket.sendto(send_data.encode('utf-8'), ("192.168.8.226",12341))
    32.     # 6、关闭套接字
    33.     udp_socket.close()
    34. def main1():
    35.     # 1、创建一个UDP套接字
    36.     udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    37.     # 2. 准备接收方的地址和端口,'192.168.0.107'表示目的ip地址,8080表示目的端口号
    38.     dest_addr = ('192.168.8.226', 12341)  # 注意这是一个元组,其中ip地址是字符串,端口号是数字
    39.     # 3. 发送数据到指定的ip和端口
    40.     for i in range(1):
    41.         udp_socket.sendto("Hello,I am a UDP socket.".encode('utf-8'), dest_addr)
    42.         time.sleep(1)
    43.     # 4. 等待接收对方发送的数据
    44.     recv_data = udp_socket.recvfrom(1024)  # 1024表示本次接收的最大字节数
    45.     # 5、打印接收到的数据
    46.     print(recv_data)
    47.     # 4. 关闭套接字
    48.     udp_socket.close()
    49. if __name__ == '__main__':
    50.     main()

    结语
    这里只是UDP的简单使用,给大家一个示例参考,在实际应用过程中,涉及到复杂数据通信,还需要使用通信协议,协议收发,解包等函数,另外数据缓存也很关键,尤其是大数据量的情况下,通常会用到队列相关知识,这一部分就留给大家自行研究吧,如果这篇文章对你有用,不妨点赞关注,你的支持是我最大的动力。

  • 相关阅读:
    【SQL】Spark SQL 比较上下两条数据,多个字段多个条件,赋值一个结果字段成一个array,
    我又学会了使用Range实现网络文件下载的断点续传
    时间轴_量子计算机
    排序算法之基数排序
    Java项目:ssm流浪狗领养系统
    【AWS实验】 使用 Lake Formation 设置数据湖
    SQL基础理论篇(一):什么是SQL
    alibaba Canal 增量订阅 & 消费组件,了解,安装,部署实践
    jdbc(DriverManager+Connection+Statement+ResultSet)+SQL注入+开启预编译+数据连接池
    架构与思维:微服务架构的思想本质
  • 原文地址:https://blog.csdn.net/xuezhe5212/article/details/140344334