OpenMV 图像串口传输示例

注意:本程序根据 OpenMV采集图片通过串口发送,PC接收并保存为图片 更改。

一、例程说明

这个例程主要实现了以下功能:

1. OpenMV 端采集图像:使用OpenMV开发板上的摄像头采集实时图像数据。

2. 通过串口传输图像数据:将采集到的图像数据打包成字节流,通过串口发送到连接的PC端。

3. PC端接收并保存图像:PC端接收来自OpenMV的图像数据,并使用OpenCV库将其解码保存为图片文件。

整个过程模拟了一个简单的图像采集-传输-保存的应用场景。OpenMV负责图像采集和数据发送,PC端负责接收和保存图像。这种基于串口通信的图像传输方式,对于一些嵌入式设备与PC之间的图像交互非常有用。

通过这个示例程序,可以学习如何在OpenMV和PC端进行串口通信,以及如何处理和保存接收到的图像数据。这对于开发基于OpenMV的图像采集和传输应用很有帮助。

使用说明:先运行PC端,再打开OpenMv。

二、硬件说明

1. STM32 ARM SWD 仿真器调试器

        我最初用的是 STM32 ARM SWD 仿真器调试器中的串口 如下图:

        用这个传输速率较慢,它不支持 921600的波特率

2. TTL

        简易使用TTL能经量减少卡顿,减少传输时间。

3. openmv

三、硬件连接

四、程序编写

4.1 OpenMv代码

 1. 导入模块

import sensor
import image
import ustruct
import pyb
from pyb import Pin

程序导入了 sensor 模块、image 模块、ustruct 模块、pyb 模块和 Pin 类。其中,sensor 模块和 image 模块是用于控制 OpenMV 摄像头的模块,ustruct 模块用于打包和解包数据,pyb 模块用于控制 OpenMV 开发板的外设,Pin 类用于控制开发板的引脚。

 2. 初始化摄像头和串口

sensor.reset() # Initialize the camera sensor.
sensor.set_pixformat(sensor.RGB565) # or sensor.RGB565
sensor.set_framesize(sensor.QVGA) # or sensor.QVGA (or others)
sensor.skip_frames(time = 2000) # Let new settings take affect.uart = pyb.UART(3, 921600)  # 选择合适的串口号和波特率

程序使用 sensor.reset() 函数初始化摄像头,并设置摄像头的像素格式为 RGB565,帧大小为 QVGA。然后,程序使用 pyb.UART() 函数初始化串口,选择合适的串口号和波特率。

注意:帧对传输速率的影响 VAG ( 图片最大,速度最慢 )

                                            QVAG ( 图片中等,速度中等 )                                           

                                            QQVAG ( 图片最小,速度最快 )

           如果 921600 波特率不行,建议减小波特率。

  3. 定义捕获和发送图像的函数

def capture_and_send_image():# 捕获图像img = sensor.snapshot()# 将图像转换为JPEG格式img_compressed = img.compress(quality=50)  #质量可以修改为 10 ~ 90# 计算图像大小size = ustruct.pack("<L", len(img_compressed))# 发送图像大小和数据uart.write(size)uart.write(img_compressed)# 等待接收到确认信号while uart.any() == 0:pass# 接收确认信号confirmation = uart.read(1)# 如果接收到的确认信号为 "#"if confirmation == b'#':# 发送停止信号 "#"uart.write(b'#')

定义一个名为 capture_and_send_image() 的函数,用于捕获图像并发送到串口。函数的主要步骤如下:

  • 使用 sensor.snapshot() 函数捕获实时图像数据。
  • 使用 img.compress() 函数将图像数据压缩为 JPEG 格式,质量为 50。
  • 使用 ustruct.pack() 函数将压缩后的图像大小打包为 4 字节的字节流。
  • 使用 uart.write() 函数将图像大小和压缩后的图像数据发送到串口。
  • 使用 uart.any() 函数等待串口接收到确认信号。
  • 使用 uart.read() 函数读取串口接收到的确认信号。
  • 如果接收到的确认信号为 #,则发送停止信号 #

  4. 主循环

while True:capture_and_send_image()

程序使用一个无限循环不断调用 capture_and_send_image() 函数,实现连续捕获和发送图像的功能。

4.2 PC端代码

主要功能是在 PC 端接收通过串口传输的图像数据,并将接收到的图像数据解码并显示出来。

  1. 导入所需的模块

from ast import Import
import cv2
import serial
import struct
import time
import numpy as np

代码导入了 ast 模块的 Import 类,cv2 模块用于图像处理和显示,serial 模块用于串口通信,struct 模块用于打包和解包数据,time 模块用于添加延时,numpy 模块用于处理图像数据。

  2. 打开串口

ser = serial.Serial('COM4', 921600)

代码使用 serial.Serial() 函数打开串口,并指定串口号和波特率。需要将 'COM4' 替换为你实际使用的串口号,当然波特率也同openmv一致。

  3. 显示图像的函数

def imshow(img):cv2.imshow("Received Image", img)cv2.waitKey(1)key = cv2.waitKey(1)if key == 32:  # 空格退出cv2.destroyAllWindows()while True:pass

这个函数用于解码并显示图像。它使用 cv2.imshow() 函数显示图像,使用 cv2.waitKey() 函数等待按键输入。如果按下空格键,窗口将关闭。

  4. 接收和保存图像的函数

def receive_and_save_image(output_path, file_preserve):size_data = ser.read(4)size = struct.unpack("<L", size_data)[0]image_data = ser.read(size)img = cv2.imdecode(np.frombuffer(image_data, dtype=np.uint8), cv2.IMREAD_COLOR)if file_preserve == 1:with open(output_path, 'wb') as file:file.write(image_data)ser.write(b'#')stop_signal = ser.read(1)if stop_signal == b'#':return img, Trueelse:return img, False

这个函数用于接收通过串口传输的图像数据,并将图像数据解码为图像。它首先读取图像大小,然后根据大小读取图像数据。接下来,使用 cv2.imdecode() 函数将图像数据解码为图像。如果 file_preserve 参数为 1,则将图像数据保存到指定的文件路径。然后,发送确认信号 # 到串口,并等待接收停止信号。如果接收到停止信号 #,则返回解码后的图像和 True;否则返回解码后的图像和 False

  5. 主循环

output_image_path = 'received_image.jpg'while True:#key = int(input("请输入: "))key = 1if key == 0:breakwhile key == 1:img, uart_img_key = receive_and_save_image(output_image_path, 0)if uart_img_key:imshow(img)break

这个循环用于不断接收和显示图像。在循环内部,调用 receive_and_save_image() 函数接收图像,并将返回的图像数据传递给 imshow() 函数进行显示。如果接收到的图像数据有效,即 uart_img_key 为 True,则退出内部循环。

这里key可以换成输入,来进行调试。

 6. 关闭串口和窗口

ser.close()
cv2.destroyAllWindows()

在循环结束后,关闭串口和窗口。

五、总代码

5.1 OpenMv

import sensor
import image
import ustruct
import pyb
from pyb import Pinsensor.reset() # Initialize the camera sensor.
sensor.set_pixformat(sensor.RGB565) # or sensor.RGB565
sensor.set_framesize(sensor.QVGA) # or sensor.QVGA (or others)
sensor.skip_frames(time = 2000) # Let new settings take affect.# 初始化相机
#sensor.reset()
#sensor.set_pixformat(sensor.RGB565)
#sensor.set_framesize(sensor.VGA)  # 分辨率可以修改为QQVGA、QVGA等
#sensor.skip_frames(time=500)# 初始化串口
#uart = pyb.UART(3, 115200)  # 选择合适的串口号和波特率
uart = pyb.UART(3, 921600)  # 选择合适的串口号和波特率# 初始化I/O
#p_in = Pin('P9', Pin.IN, Pin.PULL_UP)#设置p_in为输入引脚,并开启上拉电阻
# value = p_in.value() # get value, 0 or 1#读入p_in引脚的值# 捕获并发送图像
def capture_and_send_image():# 捕获图像img = sensor.snapshot()# 将图像转换为JPEG格式img_compressed = img.compress(quality=50)  #质量可以修改为 10 ~ 90# 计算图像大小size = ustruct.pack("<L", len(img_compressed))# 发送图像大小和数据uart.write(size)uart.write(img_compressed)# 等待接收到确认信号while uart.any() == 0:pass# 接收确认信号confirmation = uart.read(1)# 如果接收到的确认信号为 "#"if confirmation == b'#':# 停止发送图像#while uart.any():#if uart.any():#    uart.readchar()# 发送停止信号 "#"uart.write(b'#')# 主循环
while True:capture_and_send_image()  # 捕获并发送图片

5.2 PC端

#coding:GBK
from ast import Import
import cv2
import serial
import struct
import time
import numpy as np# 打开串口
ser = serial.Serial('COM4', 921600)  # 将 COM1 替换为你的串口号和相应的波特率
#ser = serial.Serial('COM3', 115200)  # 将 COM1 替换为你的串口号和相应的波特率def imshow (img):# 解码图像数据#img = cv2.imdecode(np.frombuffer(image_data, dtype=np.uint8), cv2.IMREAD_COLOR)# 显示图像cv2.imshow("Received Image", img)cv2.waitKey(1)key = cv2.waitKey(1)if key == 32: #空格退出cv2.destroyAllWindows()while True:pass# 接收图像并保存
#file_preserve = 1 保存图片文件
def receive_and_save_image(output_path, file_preserve):# 读取图像大小size_data = ser.read(4)size = struct.unpack("<L", size_data)[0]# 读取图像数据image_data = ser.read(size)# 解码图像数据img = cv2.imdecode(np.frombuffer(image_data, dtype=np.uint8), cv2.IMREAD_COLOR)# 保存图像if file_preserve == 1:with open(output_path, 'wb') as file:file.write(image_data)# 发送确认信号 "#"ser.write(b'#')# 接收停止信号stop_signal = ser.read(1)if stop_signal == b'#':#imshow(image_data)return img,Trueelse:return img,False# 图像保存路径
output_image_path = 'received_image.jpg'while True:# 在需要延时的地方调用sleep()函数#time.sleep(0.01)  # 延时0.1秒# 循环接收并保存图像#key = int(input("请输入: "))key = 1if key == 0:breakwhile key == 1:img, uart_img_key = receive_and_save_image(output_image_path, 0, 0)if uart_img_key:imshow(img)break# 关闭串口
ser.close()
cv2.destroyAllWindows()

六、相关资料

1. OpenMV采集图片通过串口发送,PC接收并保存为图片

2. OpenMv详细参数

3. Visual Studio Installer 安装python库

4. Visual Studio Installer 运行python 汉字

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

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

相关文章

idea 项目 修改项目文件名 教程

文章目录 目录 文章目录 修改流程 小结 概要流程技术细节小结 概要 原项目名 修改流程 关掉当前项目的idea页面 修改之后的文件名 重新打开idea。选择项目打开项目页面 技术细节 出现下面这个问题&#xff0c;可以参考作者新的一编文章idea开发工具 项目使用Spring框架开发解…

【Linux】25. 网络基础(一)

网络基础(一) 计算机网络背景 网络发展 独立模式: 计算机之间相互独立; 网络互联: 多台计算机连接在一起, 完成数据共享; 其实本质上一台计算机内部也是一个小型网络结构(如果我们将计算机内部某个硬件不存放在电脑中&#xff0c;而是拉根长长的线进行连接。这其实也就是网…

在做题中学习(51): x的平方根

69. x 的平方根 - 力扣&#xff08;LeetCode&#xff09;​​​​​​ 解法&#xff1a;二分查找 思路&#xff1a;看示例2&#xff1a; 可以看到8的平方根是2.82&#xff0c;在2^2和3^2之间&#xff0c;所以可以把数组分为两部分&#xff0c;(具有二段性) 而2.82去掉小数部分…

【Kolmogorov-Arnold网络 替代多层感知机MLPs】KAN: Kolmogorov-Arnold Networks

KAN: Kolmogorov-Arnold Networks 论文地址 代码地址 知乎上的讨论&#xff08;看一下评论区更正&#xff09; Abstract Inspired by the Kolmogorov-Arnold representation theorem, we propose Kolmogorov-Arnold Networks (KANs) as promising alternatives to Multi-Layer…

【OceanBase 系列】—— OceanBase v4.3 特性解读:查询性能提升之利器列存储引擎

原文链接&#xff1a;OceanBase 社区 对于分析类查询&#xff0c;列存可以极大地提升查询性能&#xff0c;也是 OceanBase 做好 HTAP 和 OLAP 的一项不可缺少的特性。本文介绍 OceanBase 列存的实现特色。 OceanBase从诞生起就一直坚持LSM-Tree架构&#xff0c;不断打磨功能支…

2020 年国考【计算机专业】真题及答案

真题及答案 第一部分数学基础课程 一、&#xff08;共 5 分&#xff09;用逻辑符号表达下列语句&#xff08;论域为包含一切事物的合集&#xff09; &#xff08;2 分&#xff09;确诊者并不都有症状&#xff08;注&#xff1a;需给出两种形式表达, 一种用存在量词, 一种用全称…

python魔法方法是什么

魔法方法是python内置方法&#xff0c;不需要主动调用&#xff0c;存在的目的是为了给python的解释器进行调用&#xff0c;几乎每个魔法方法都有一个对应的内置函数&#xff0c;或者运算符&#xff0c;当我们对这个对象使用这些函数或者运算符时就会调用类中的对应魔法方法&…

MVC与MVVM架构模式

1、MVC MVC&#xff1a;Model-View-Controller&#xff0c;即模型-视图-控制器 MVC模式是一种非常经典的软件架构模式。从设计模式的角度来看&#xff0c;MVC模式是一种复合模式&#xff0c;它将多个设计模式结合在一种解决方案中&#xff0c;从而可以解决许多设计问题。 MV…

【3D目标检测】常见相关指标说明

一、mAP指标 mean Average Precision&#xff08;平均精度均值&#xff09;&#xff0c;它是目标检测和信息检索等任务中的重要性能指标。mAP 通过综合考虑精度和召回率来衡量模型的总体性能。 1.1 精度&#xff08;Precision&#xff09; 表示检索到的目标中实际为正确目标…

Spring Task及订单状态定时处理

1&#xff1a;Spring Task概念&#xff1a; Spring Task 是Spring框架提供的任务调度工具&#xff0c;可以按照约定的时间自动执行某个代码逻辑 定时任务的理解 定时任务即系统在特定时间执行一段代码&#xff0c;它的场景应用非常广泛&#xff1a; 购买游戏的月卡会员后&a…

Centos7网络处理name or service not known

1、编辑->虚拟网络编辑器 2、查看本机的ip 3、 /etc/sysconfig/network-scripts/ 查看文件夹下面的 ifcfg-eth33 后面的33可能不一样 vi /etc/resolv.conf 编辑文件添加以下DNS nameserver 114.114.114.114 4、设置本机的网络 5、ping www.baidu.com 先重启…

第50期|GPTSecurity周报

GPTSecurity是一个涵盖了前沿学术研究和实践经验分享的社区&#xff0c;集成了生成预训练Transformer&#xff08;GPT&#xff09;、人工智能生成内容&#xff08;AIGC&#xff09;以及大语言模型&#xff08;LLM&#xff09;等安全领域应用的知识。在这里&#xff0c;您可以找…

全双工音频对讲模块-支持空中升级、多级无线中继

SA618F30是一款高集成的大功率全双工无线音频模块&#xff0c;发射功率高达32dBm。该音频模块简化接口&#xff0c;只需外接音频功放或麦克风即可作为一个小型对讲机&#xff0c;方便快捷嵌入到各类手持设备中。支持多级无线中继&#xff0c;支持OTA空中升级。 SA618F30配备1W…

Java快速入门系列-11(项目实战与最佳实践)

第十一章&#xff1a;项目实战与最佳实践 11.1 项目规划与需求分析项目规划需求分析实例代码 11.2 系统设计考虑实例代码 11.3 代码实现与重构实例代码 11.4 性能优化与监控实例代码 11.5 部署与持续集成/持续部署(CI/CD)实例代码 11.1 项目规划与需求分析 在进行任何软件开发…

Hikyuu-PF-银行股轮动交易策略实现

今天&#xff0c;带来的是“如何使用 Hikyuu 中的投资组合来实现银行股轮动交易策略”。 这个策略的逻辑很简单&#xff1a;持续持有两支市净率最低银行股&#xff0c;然后每月换仓 定义回测周期与回测标的 同样&#xff0c;首先定义回测周期&#xff1a; # 定义回测日期 …

了解内存函数

✨✨欢迎&#x1f44d;&#x1f44d;点赞☕️☕️收藏✍✍评论 个人主页&#xff1a;秋邱博客 所属栏目&#xff1a;C语言 前言 内存函数不止malloc、calloc、realloc、free还有memcpy、memmove、memset、memcmp。前四个的头文件是<stdlib.h>,后四个的头文件是<strin…

Ansible----playbook模块之templates模块、tags模块、roles模块

目录 引言 一、templates模块 &#xff08;一&#xff09;关键信息 &#xff08;二&#xff09;实际操作 1.定义主机组 2.设置免密登录 3.分别建立访问目录 4.定义模板文件 5.创建playbook文件 6.执行剧本 7.验证结果 二、tags模块 &#xff08;一&#xff09;创建…

《QT实用小工具·六十一》带动画的三角形指示箭头

1、概述 源码放在文章末尾 该项目实现了一个带动画效果的三角形指示箭头&#xff0c;项目demo演示如下所示&#xff1a; 用法 interestingindicate.h interestingindicate.cpp 放到工程中&#xff0c;直接使用即可。 注意&#xff1a;建议绝对布局&#xff0c;手动指定 wid…

华为车BU迈入新阶段,新任CEO对智能车的3个预判

作者 |张马也 编辑 |德新 4月24日&#xff0c;北京车展前夕&#xff0c;华为召开了新一年的智能汽车解决方案新品发布会。 这次发布会&#xff0c;也是华为智能汽车解决方案BU&#xff08;简称「车BU」&#xff09;CEO 靳玉志的公开首秀。 一开场&#xff0c;靳玉志即抛出了…

BACnet转MQTT网关智联楼宇json格式自定义

智能建筑的BACnet协议作为楼宇自动化领域的通用语言&#xff0c;正逐步迈向更广阔的物联网世界。随着云计算和大数据技术的飞速发展&#xff0c;如何将BACnet设备无缝融入云端生态系统&#xff0c;成为众多楼宇管理者关注的焦点。本文将以一个实际案例&#xff0c;揭示BACnet网…