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, True
    else:
        return img, False

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

  5. 主循环

output_image_path = 'received_image.jpg'

while True:
    #key = int(input("请输入: "))
    key = 1
    if key == 0:
        break
    
    while 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 Pin


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.

# 初始化相机
#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,True
    else:
        return img,False

# 图像保存路径
output_image_path = 'received_image.jpg'


while True:
    # 在需要延时的地方调用sleep()函数
    #time.sleep(0.01)  # 延时0.1秒
    # 循环接收并保存图像
    #key = int(input("请输入: "))
    key = 1
    if key == 0:
        break
    
    while 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 汉字

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

智慧工地的5大系统是什么?SaaS化大型微服务架构(智慧工地云平台源码)可多端展示登录

智慧工地解决方案依托计算机技术、物联网、云计算、大数据、人工智能、VR&AR等技术相结合&#xff0c;为工程项目管理提供先进技术手段&#xff0c;构建工地现场智能监控和控制体系&#xff0c;弥补传统方法在监管中的缺陷&#xff0c;最终实现项目对人、机、料、法、环的全…

外企接受大龄程序员吗?

本人知乎账号同公众号&#xff1a;老胡聊Java&#xff0c;欢迎留言并咨询 亲身体会外企经历所见所闻&#xff0c;外企能接受大龄程序员。 1 大概是10年的时候&#xff0c;进一家知名外企&#xff0c;和我一起进的一位manager&#xff0c;后来听下来&#xff0c;年龄35&#xf…

html的标签

基础标签 标签描述<h1>-<h6>定义标题&#xff0c;h1最大&#xff0c;h6最小<font>定义文本的字体&#xff0c;字体尺寸&#xff0c;字体颜色<b>定义粗体文本<i>定义斜体文本<u>定义文本下划线<center>定义文本居中<p>定义段落…

俄罗斯国际消费类电子电器展ICEE:人潮如织,盛况空前

近日&#xff0c;备受全球瞩目的俄罗斯国际消费类电子电器展ICEE在莫斯科盛大落幕。本次展会为期四天&#xff0c;真的攒足了眼球&#xff0c;不仅俄罗斯这边的很多媒体和自媒体有报道&#xff0c;展会第一天&#xff0c;很多参展商通过短视频平台将展会的盛况传到了国内&#…

Ubuntu22.04下安装kafka_2.11-0.10.1.0并运行简单实例

目录 一、版本信息 二、安装Kafka 1.将Kafka安装包移到下载目录中 2.下载Spark并确保hadoop用户对Spark目录有操作权限 三、启动Kafka并测试Kafka是否正常工作 1.启动Kafka 2.测试Kafka是否正常工作 一、版本信息 虚拟机产品&#xff1a;VMware Workstation 17 Pro 虚…

AI PC,到底谁的梦想、谁的红利?

世界上第一台通用计算机是1946年诞生的埃尼阿克&#xff0c;世界上第一台便携电脑是1981年诞生的“奥斯本1号”&#xff0c;世界上第一部智能手机是1993年诞生的IBM Simon。 它们之间看起来毫无关系&#xff0c;但却暗含一种关于创新的微妙潜规则——随着时间推移&#xff0c;…

MM模块学习一(供应商创建,物料类型的定义及功能)

物料管理流程&#xff1a; 源头&#xff1a;采购需求->采购申请 MRP&#xff1a;物料需求计划。运行物料需求计划的结果&#xff0c;根据物料的性质来判断是外购&#xff08;采购申请&#xff09;或者是生产&#xff08;计划订单->生产订单&#xff09;。 采购申请&am…

acwing算法提高之数据结构--平衡树Treap

目录 1 介绍2 训练 1 介绍 本博客用来记录使用平衡树求解的题目。 插入、删除、查询操作的时间复杂度都是O(logN)。 动态维护一个有序序列。 2 训练 题目1&#xff1a;253普通平衡树 C代码如下&#xff0c; #include <cstdio> #include <cstring> #include …

Postgresql源码(128)深入分析JIT中的函数内联llvm_inline

相关 《Postgresql源码&#xff08;127&#xff09;投影ExecProject的表达式执行分析》 《LLVM的ThinLTO编译优化技术在Postgresql中的应用》 《LLVM&#xff08;5&#xff09;ORC实例分析》 1 JIT优化效果 create table t1(i int primary key, j int, k int); insert into t1…

嵌入式开发三:STM32初体验

本节主要向大家介绍如何开发过程中的基本操作&#xff0c;如编译、串口下载、仿真器下载、仿真调试程序&#xff0c;体验一下 STM32 的开发流程&#xff0c;并介绍 MDK5 的一些使用技巧&#xff0c;通过本节的学习&#xff0c;将对 STM32 的开发流程和 MDK5 使用有个大概了解&a…

Windows安全加固-账号与口令管理

在当今日益增长的网络安全威胁中&#xff0c;Windows系统的安全加固显得尤为重要。其中&#xff0c;账号与口令管理作为系统安全的第一道防线&#xff0c;其重要性不言而喻。本文将深入探讨Windows安全加固中的账号与口令管理策略&#xff0c;以确保系统的安全性和稳定性。 账…

认识大模型提示词

一、写作助理 &#x1f4a5;最常使用的 prompt&#xff0c;用于优化文本的语法、清晰度和简洁度&#xff0c;提高可读性。 输入&#xff1a;作为一名写作改进助理&#xff0c;你的任务是改进所提供文本的拼写、语法、清晰、简洁和整体可读性&#xff0c;同时分解长句&#xff…

运行SpringBoot项目失败?异常显示Can‘t load IA 32-bit .dll on a AMD 64-bit platform,让我来看看~

原因是&#xff0c;我放入jdk的bin文件夹下的tcnative-1.dll文件是32位的&#xff0c;那么肯定是无法在AMD 64位平台上加载IA 32位.dll。但是网站上给出的都是32位呀&#xff0c;没有64位怎么办&#xff1a; 其实当我们把“tomcat-native-1.2.34-openssl-1.1.1o-win32-bin.zip”…

项目管理-项目资源管理2/2

项目管理&#xff1a;每天进步一点点~ 活到老&#xff0c;学到老 ヾ(◍∇◍)&#xff89;&#xff9e; 何时学习都不晚&#xff0c;加油 资源管理&#xff1a;6个过程“硅谷火箭管控” ①规划资源管理&#xff1a; 写计划 ②估算活动资源&#xff1a;估算团队资源&…

数据结构学习——二叉树

1. 树概念及结构 1.1 树的概念 树是一种非线性的数据结构&#xff0c;它是由n&#xff08;n>0&#xff09;个有限结点组成一个具有层次关系的集合。把它叫做树是因为它看起来像一棵倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;而叶朝下的。 有一个特殊的结点&…

C语言:指针(1)

1. 内存和地址 内存划分为⼀个个的内存单元&#xff0c;每个内存单元的⼤⼩取1个字节。 计算机中常⻅的单位&#xff08;补充&#xff09;&#xff1a; ⼀个⽐特位可以存储⼀个2进制的位1或者0 C语⾔中给地址起了新的名字叫&#xff1a;指针。 内存单元的编号地址指针。 1.…

实战BACnet/IP标准通信网关在楼宇自动化中的应用

智慧楼宇建设实现不同设备间的互联互通是一项巨大挑战&#xff0c;尤其是在那些历史悠久的建筑中&#xff0c;新旧系统并存的情况尤为普遍。某大型商业综合体就面临着这样的困境&#xff1a;老旧的暖通空调系统采用Modbus RTU协议&#xff0c;而新部署的能源管理系统却要求BACn…

企业车辆管理系统参考论文(论文 + 源码)

【免费】关于企业车辆管理系统.zip资源-CSDN文库https://download.csdn.net/download/JW_559/89282550 企业车辆管理系统 摘 要 随着经济的日益增长,车辆作为最重要的交通工具,在企事业单位中得以普及,单位的车辆数目已经远远不止简单的几辆,与此同时就产生了车辆资源的合理…

react状态管理之state

第三章 - 状态管理 随着你的应用不断变大&#xff0c;更有意识的去关注应用状态如何组织&#xff0c;以及数据如何在组件之间流动会对你很有帮助。冗余或重复的状态往往是缺陷的根源。在本节中&#xff0c;你将学习如何组织好状态&#xff0c;如何保持状态更新逻辑的可维护性&…

Java特性之设计模式【代理模式】

一、代理模式 概述 在代理模式&#xff08;Proxy Pattern&#xff09;中&#xff0c;一个类代表另一个类的功能。这种类型的设计模式属于结构型模式 在代理模式中&#xff0c;我们创建具有现有对象的对象&#xff0c;以便向外界提供功能接口 主要解决&#xff1a; 在直接访问…