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

news2024/9/8 19:26:02

简介
在网络通信编程中,用的最多的就是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、创建一个UDP套接字
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

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

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

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

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

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

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

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

#!/usr/bin/python3
# -*- coding: utf-8 -*-

"""
udp通信例程:udp server端,修改udp_addr元组里面的ip地址,即可实现与目标机器的通信,
此处以单机通信示例,ip为127.0.0.1,实际多机通信,此处应设置为目标客户端ip地址
"""

from time import sleep
import socket


def main():
    # udp 通信地址,IP+端口号
    udp_addr = ('127.0.0.1', 9999)
    udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    # 绑定端口
    udp_socket.bind(udp_addr)

    # 等待接收对方发送的数据
    while True:
        recv_data = udp_socket.recvfrom(1024)  # 1024表示本次接收的最大字节数
        # 打印接收到的数据
        print("[From %s:%d]:%s" % (recv_data[1][0], recv_data[1][1], recv_data[0].decode("utf-8")))

if __name__ == '__main__':
    print("当前版本: ", __version__)
    print("udp server ")
    main()

代码解析 

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

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

#!/usr/bin/python3
# -*- coding: utf-8 -*-

"""
udp通信例程:udp client端,修改udp_addr元组里面的ip地址,即可实现与目标机器的通信,
此处以单机通信示例,ip为127.0.0.1,实际多机通信,此处应设置为目标服务端ip地址
"""

from time import sleep
import socket

def main():
    # udp 通信地址,IP+端口号
    udp_addr = ('127.0.0.1', 9999)
    udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

    # 发送数据到指定的ip和端口,每隔1s发送一次,发送10次
    for i in range(10):
        udp_socket.sendto(("Hello,I am a UDP socket for: " + str(i)) .encode('utf-8'), udp_addr)
        print("send %d message" % i)
        sleep(1)

    # 5. 关闭套接字
    udp_socket.close()


if __name__ == '__main__':
    print("当前版本: ", __version__)
    print("udp client ")
    main()


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

#!/usr/bin/python3
# -*- coding: utf-8 -*-

"""
python多线程通信
"""

from time import sleep
import socket
import threading

# 定义全局变量
t1_count = 0
t2_count = 0


def udp_received_hundle(s):
    global t1_count
    print("this is thread 1 running")
    while True:
        t1_count += 1
        print("thread 1 第 %s 次运行" % t1_count)
        recv_data = s.recvfrom(1024)  # 1024表示本次接收的最大字节数
        # 打印接收到的数据
        print("[From %s:%d]:%s" % (recv_data[1][0], recv_data[1][1], recv_data[0].decode("utf-8")))


def udp_send_hundle(s):
    global t2_count
    print("this is thread 2 running")
    while True:
        t2_count += 1
        print("")
        print("thread 2 第 %s 次运行" % t2_count)
        s.sendto(("Hello,I am a UDP socket for: " + str(t2_count)).encode('utf-8'), udp_addr)
        print("send %d message" % t2_count)
        print("")
        sleep(1)


if __name__ == '__main__':
    print("当前版本: ", __version__)
    # 初始化
    # udp 通信地址,IP+端口号
    udp_addr = ('127.0.0.1', 9999)
    udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    # 绑定端口:
    udp_socket.bind(udp_addr)

    # 定义线程
    thread_list = []
    t1 = threading.Thread(target=udp_received_hundle, args=(udp_socket, ))
    thread_list.append(t1)
    t2 = threading.Thread(target=udp_send_hundle, args=(udp_socket, ))
    thread_list.append(t2)

    for t in thread_list:
        t.setDaemon(True)
        t.start()
    for t in thread_list:
        t.join()

    print("exit all task.")
    print('all process end.')

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

当前版本:  1.0.0
this is thread 1 running
thread 1 第 1 次运行
this is thread 2 running

thread 2 第 1 次运行
send 1 message

[From 127.0.0.1:9999]:Hello,I am a UDP socket for: 1
thread 1 第 2 次运行

thread 2 第 2 次运行
send 2 message

[From 127.0.0.1:9999]:Hello,I am a UDP socket for: 2
thread 1 第 3 次运行

thread 2 第 3 次运行
send 3 message

[From 127.0.0.1:9999]:Hello,I am a UDP socket for: 3
thread 1 第 4 次运行

thread 2 第 4 次运行
send 4 message

[From 127.0.0.1:9999]:Hello,I am a UDP socket for: 4
thread 1 第 5 次运行

thread 2 第 5 次运行
send 5 message

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

完整代码:

​#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author: William

import socket,time

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

    # 2. 准备接收方的地址和端口,'192.168.0.107'表示目的ip地址,8080表示目的端口号
    dest_addr = ('192.168.8.226', 12341)  # 注意这是一个元组,其中ip地址是字符串,端口号是数字

    # 3. 发送数据到指定的ip和端口
    for i in range(10):
        udp_socket.sendto("Hello,I am a UDP socket.".encode('utf-8'), dest_addr)
        time.sleep(1)

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

if __name__ == '__main__':
    main()

​

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

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

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

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

接收完后将其打印出来:

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

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

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

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author: William

import socket,time

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

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

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

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

    # 5. 关闭套接字
    udp_socket.close()

if __name__ == '__main__':
    main()


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

完成代码:

​
#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author: William

import socket,time

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

    # 2. 绑定本地的相关信息,如果不绑定,则系统会随机分配一个端口号
    udp_socket.bind(('', 12344))

    # 3. 发送数据到指定的ip和端口,每隔1s发送一次,发送10次
    for i in range(10):
        udp_socket.sendto("Hello,I am a UDP socket.".encode('utf-8'), ('192.168.8.226', 12341))
        time.sleep(1)

    # 4. 等待接收对方发送的数据
    while(True):
        recv_data = udp_socket.recvfrom(1024)
        # 打印接收到的数据
        print("[From %s:%d]:%s"%(recv_data[1][0],recv_data[1][1],recv_data[0].decode("utf-8")))

    # 5. 关闭套接字
    udp_socket.close()

if __name__ == '__main__':
    main()

​


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

实现代码:

​
#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author: William

import socket,time,threading

def recv_thread(socket):
    # 等待接收对方发送的数据
    while(True):
        try:
            recv_data = socket.recvfrom(1024)
            # 打印接收到的数据
            print("[From %s:%d]:%s"%(recv_data[1][0],recv_data[1][1],recv_data[0].decode("utf-8")))
        except Exception:
            break

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

    # 2、绑定本地的相关信息,如果不绑定,则系统会随机分配一个端口号
    udp_socket.bind(('', 12344))

    # 3、打印本机ip地址和端口号
    print("local ipaddr and port->",socket.gethostbyname(socket.gethostname())+":12344")

    # 4、创建一个线程,用来接收数据
    t = threading.Thread(target=recv_thread, args=(udp_socket,))
    t.start()

    # 5、等待输入数据,然后发送出去,直到输入的数据为'quit'
    while(True):
        print("please input a string.input 'quit' to quit.")
        send_data = input()
        if send_data == "quit":
            break
        else:
            udp_socket.sendto(send_data.encode('utf-8'), ("192.168.8.226",12341))

    # 6、关闭套接字
    udp_socket.close()

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

    # 2. 准备接收方的地址和端口,'192.168.0.107'表示目的ip地址,8080表示目的端口号
    dest_addr = ('192.168.8.226', 12341)  # 注意这是一个元组,其中ip地址是字符串,端口号是数字

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

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

    # 5、打印接收到的数据
    print(recv_data)
    # 4. 关闭套接字
    udp_socket.close()

if __name__ == '__main__':
    main()


​

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1918270.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

dolphinScheduler + hive + datax报错记录

1、参数错误 报错信息 [INFO] 2024-04-11 06:43:18.386 - [taskAppIdTASK-29-3301-84461]:[498] - after replace sql , preparing : insertoverwrite table mis_month partition (dt) select nvl(sl.slid , ) as id,--水量 IDnvl(sl.hh …

【CORS 报错】跨域请求问题:CORS 多种环境下的解决方案

🔥 个人主页:空白诗 文章目录 一、CORS错误的常见原因二、解决方案1. Vue3 Vite项目下的解决方案创建Vue3 Vite项目配置Vite的代理发送请求 2. jQuery项目下的解决方案使用CORS请求头使用JSONP 3. 其他环境下的解决方案使用服务器端代理设置CORS头使用…

使用django-haystack+whoosh实现全文搜索

前言 好像是上个星期在写代码的时候遇到了一些问题,这个问题似乎我之前也遇到过,印象中好像也写博客进行记录了的,于是就想在我的博客系统中“查找”(表示很无奈居然没有搜索功能🥹🥹)&#xff…

python--实验 9 类和对象

知识点 面向对象编程思想 面向对象编程是一种编程范式,它使用“对象”来设计软件,这些对象可以包含数据和代码,即属性和方法。 面向对象的程序设计的核心是对象,世间存在的万物皆为对象(上帝式思维) 面向对象&#xf…

科普文:一天学会shell编程

1.shell概叙 本文将从shell执行、语法、实战三个方面来讲解shell编程,其实shell编程就是个批处理,将你平时在服务器上单独执行的命令,按照一定要求组织起来,写在一起,然后统一执行,就完事了。 对于运维人员…

零基础小白人手必备“新华字典”,涵盖入门到进阶,超全语法!精炼总结64页,背完你的Python就牛了!

《Python背记手册》是一本旨在帮助初学者快速入门Python的实用手册,其特点鲜明,内容全面且易于理解。 书籍PDF已打包好,戳这里领取 一、书籍基本信息 作者:书籍的作者通常具有深厚的Python开发背景,有的作者是在美国攻…

MAVSDK示例takeoff_and_land编译与使用

1.克隆MAVSDK源码 ,示例位于源码的examples中 2.安装MAVSDK,下载https://github.com/mavlink/MAVSDK/releases/download/v2.12.2/mavsdk-windows-x64-release.zip 下载后解压:

三星首款智能戒指 Galaxy Ring 将于7月24日上市,售价399美元

三星电子即将推出其首款智能戒指——Galaxy Ring,这款创新的可穿戴设备将于7月24日上市,定价为399美元。Galaxy Ring的亮相标志着三星在智能穿戴设备领域的新突破,它不仅证明了三星在技术革新上的持续能力,也显示了其在健康和健身…

编号根据规则自增生成,及spring事务和锁

1、 背景 需要根据一些规则来生成自增编号,比如:95JS0001,950002 95JS是固定的,而后缀的0001的长度也是可配置的,因为有一张表来进行维护 CREATE TABLE number_control (id bigint NOT NULL AUTO_INCREMENT COMMENT 主…

DW03D是一款用于锂离子/聚合物电池保护的高集成度解决方案。DW03D包含内部功率MOSFET、高精度电压检测电路和延迟电路

一般概述 DW03D产品是单节锂离子/锂聚合物可充电电池组保护的高集成度解决方案。DW03D包括了先进的功率MOSFET,高精度的电压检测电路和延时电路。 DW03D具有非常小的TSS08-8的封装,这使得该器件非常适合应用于空间限制得非常小的可充电电池组应用。…

生信软件25 - 三代测序数据灵敏比对工具ngmlr

1. ngmlr简介 CoNvex Gap-cost Ments for Long Reads(ngmlr)是一种长reads比对工具,可以将PacBio或Oxford Nanopore灵敏地与(大)参考基因组(比如人类参考基因组)对齐,能快速和正确地…

OpenSSH Server远程代码执行漏洞 (CVE-2024-6387)|centos7升级到最新版本OpenSSH-9.8.p1

一、漏洞概述 漏洞名称 OpenSSH Server远程代码执行漏洞 CVE ID CVE-2024-6387 漏洞类型 竞争条件 发现时间 2024-07-01 漏洞评分 暂无 漏洞等级 高危 攻击向量 网络 所需权限 无 利用难度 高 用户交互 无 PoC/EXP 未公开 在野利用 未发现 OpenS…

MechMind结构光相机 采图SDK python调用

测试效果 Mech-Mind结构光相机 Mech Mind(梅卡曼德)的结构光相机,特别是Mech-Eye系列,是工业级的高精度3D相机,广泛应用于工业自动化、机器人导航、质量检测等多个领域。以下是对Mech Mind结构光相机的详细解析&#…

阿里巴巴国际站携手NBA传奇托尼·帕克,中国卖家又一波利好!

在全球化浪潮日益汹涌的今天,跨界合作已成为推动品牌国际化进程的重要力量。近日,阿里巴巴国际站宣布了一项震撼业界的合作——正式签约NBA(美国职业篮球联赛)传奇控球后卫托尼帕克(Tony Parker)作为其全球…

2024年10款免费的项目管理软件推荐

本文向大家推荐10款2024年免费使用的项目管理软件,其中包括桌面应用和基于Web平台的多种产品,同时还涵盖了一些优秀的开源软件。 1.禅道开源项目管理软件 禅道是一款开源的、基于Web的项目管理软件,其功能丰富且操作简便,为团队提…

生产管理系统功能全拆解:哪些功能是企业真正需要的?

制造业的伙伴经常听到“生产管理”,但很多人可能只是模糊地知道它与工厂、生产线有关。那么,到底什么是生产管理呢?它的重要性又体现在哪里呢?接下来,我就以轻松的方式,带大家走进生产管理的世界&#xff0…

笔记:Qt开发之工程的多模块设计(pri)

目标:对于功能模块较多的Qt项目,使用pri文件管理模块文件,降低工程复杂度,提高软件模块的封装性和重用性。 一、知识储备 1.1 pro与pri文件 对于模块化编程,Qt提供了pro和pri,pro管理项目,pri…

算法 —— 模拟

目录 替换所有的问号 提莫攻击 Z字形变换 外观数列 各位读者有听说过“建模”一词吗?所谓“建模”,就是把事物进行抽象,根据实际问题来建立对应的数学模型。“抽象”并不意味着晦涩难懂;相反,它提供了大量的便利。…

新兴商业模式如何破局?市场策划专家的实战指南

在这个融合了传统市场、互联网和信息技术的大潮中,想要在市场营销策划上玩得转,咱们得有超凡的全局思维和跨界的协作精神。 下面,我就来和大家聊聊如何在这样一个复杂环境下搞定市场营销策划,让你在竞争激烈的市场中脱颖而出。 …

Ubuntu编译PX4固件

目录 前言 准备编译参考 前言 要想自己编译PX4固件需要交叉编译器,交叉编译器可以将 x86架构 平台上写好程序编译出来,而编译出来的可执行文件是能用到 arm架构 的平台上。 本次编译是以 px4 v1.13.2 为例。 我的配置如下: 虚拟机 Ubuntu 18…