1、什么是光流传感器
光流传感器是一种用于测量物体相对于周围环境的运动的设备。它通过检测周围光线的变化来计算出物体的运动方向和速度,广泛应用于机器人导航、无人机飞行控制、虚拟现实等领域。
2、光流传感器是如何工作的
光流是指在连续图像序列中物体表面上的像素位移,它是由于物体在图像平面上的运动而引起的。它通过比较相邻图像帧中像素位置的变化,可以计算出每个像素点的光流向量,表示该点在图像中的运动方向和速度,从而用来识别运动物体、测量运动速度、估计相机运动,飞行导航等应用。
光流传感器基于光学原理工作,它利用相机获取连续多帧图像,并通过比较相邻帧之间的像素位移来计算物体运动的速度和方向。当物体移动时,相邻帧中的像素会产生位移,通过追踪这些位移,光流传感器能够准确地捕捉物体的运动信息。
通过给图像的每一个像素点赋予一个速度矢量,这就形成一个图像运动场, 图像上的点与三维物体的点一一对应,根据各个像素点的矢量特征,可以对图像运动进行状态分析。
如果图像中没有运动物体,则光流矢量在整个图像区域的变化是连续平坦的。当图像中有运动物体时,物体和图像背景存在相对运动,运动物体的速度矢量必然和相邻背景的速度矢量不同,从而检测出运动物体及其位置。
光流传感器的一个主要应用就是自动驾驶。在自动驾驶中,光流技术主要用于传感器融合和物体追踪,以提高车辆对周围环境的感知能力。通过光流技术,可将相机和雷达传感器的数据融合,提供高分辨率的图像信息和距离和速度等信息。使得车辆能够更好地理解周围环境。
光流技术在无人机中的应用也非常广泛,包括姿态控制:通过光流传感器可获取其在空中的位置和姿态信息;路径规划和避障;通过计算光流向量,可以确定无人机与障碍物的距离和方向,从而实现避障和导航操作。
光流传感器技术经过多年的发展,已经实现了从单点测量到多点测量、从二维空间到三维空间的扩展。通过结合多摄像头、激光雷达等传感器,光流传感器能够实现对三维空间中物体运动的准确测量,在机器人导航、无人驾驶、虚拟现实等领域具有广阔的应用前景。
3、光流传感器实验演示
最后,我们来演示如何使用配备显示屏的树莓派读取并显示光流传感器的数据。实验中使用的是一款常见的光流传感器。我们将手放在传感器的上方,做前后左右的移动,可以看到,屏幕上标记的两个轴的坐标值以及绿色方向指示柱的长度,会出现相应的增大或减小的变化。
主控板卡:基于RP2040的带屏调试助手
以下是主程序中对于光流传感器数据的获取代码示例:
from breakout_colourlcd240x240 import BreakoutColourLCD240x240
from machine import ADC, Pin, Timer, PWM,I2C, SPI
from utime import sleep
import time, math,array,struct
import pmw3901
#------------------------------------------------------------------
######################################################################
pmw_cs = Pin(25)
pmw_sck=Pin(26)
pmw_mosi=Pin(27)
pmw_miso=Pin(28,Pin.IN,Pin.PULL_DOWN)
spi0=SPI(1,baudrate=400_000, phase=1, polarity=1, sck=pmw_sck, mosi=pmw_mosi,miso=pmw_miso)
PMW = pmw3901.PMW3901(spi0,pmw_cs)
Value = [0,0]
######################################################################
#初始化
#显示屏
width = BreakoutColourLCD240x240.WIDTH
height = BreakoutColourLCD240x240.HEIGHT
display_buffer = bytearray(width * height*2)
display = BreakoutColourLCD240x240(display_buffer)
current_temp_Y = 0
current_temp_X = 0
#-------------------------------------------------------------------
#屏幕基本图形绘制
def display_init():
#绘制Y轴计
display.set_pen(0,255,0)
display.rectangle(30,30,13,160)
display.circle(36,190,6)
display.set_pen(255,0,0)
#绘制X轴计
display.set_pen(0,255,0)
display.rectangle(36,184,160,13)
display.circle(64,190,6)
display.set_pen(255,0,0)
#绘制文字
display.text("current", 150, 20, 194, 2)
display.text("Y_axis", 150, 35, 194, 2)
display.text("current", 150, 100, 194, 2)
display.text("X_axis", 150, 115, 194, 2)
display.update()
display.update()
#---------------------------------------------------------------------
def display_change(temp, color):
global current_temp_Y,current_temp_X
current_temp_Y = temp[0]
current_temp_X = temp[1]
#print(temp)
display.set_pen(color[0], color[1], color[2])
display.rectangle(30,30,13,160)
# display.circle(64,190,6)
display.set_pen(0,0,150)
display.rectangle(30,20,13,7+int((25-temp[0]/2)/2)*6)
display.set_pen(0,0,0)
display.rectangle(145,50,100,40)
display.set_pen(0,255,0)
display.text(str(temp[0]), 145, 50, 5, 5)
# display.update()
display.set_pen(color[0], color[1], color[2])
display.rectangle(36,184,160,13)
display.circle(36,184,6)
display.set_pen(0,0,150)
display.rectangle(7+int((50-temp[1]/2)/2)*6,184,160,13)
display.set_pen(0,0,0)
display.rectangle(145, 130, 100, 40)
display.set_pen(0,255,0)
display.text(str(-temp[1]), 145, 130, 5, 5)
display.update()
#----------------------------------------------------------------------
def get_temp():
global Value
if PMW._read(pmw3901.REG_DATA_READY ) & 0x80 :
shutter_up = PMW._read(0x0C)
squal = PMW._read(0x07);
if shutter_up < 15 and squal >100 :
deltaX = (PMW._read(0x04) << 8) | PMW._read(0x03)
deltaY = (PMW._read(0x06) << 8) | PMW._read(0x05)
if deltaX > 32767 :
deltaX -= 65535
if deltaY > 32767 :
deltaY -= 65535
Value[0] = (Value[0]+deltaX)/10.0
Value[1] = (Value[1]+deltaY)/10.0
return Value
#-----------------------------------------------------------------------
def main():
global current_temp_Y,current_temp_X
timer1 = Timer()
color = [0,255,0]
display_init()
#timer初始化
timer1.init(freq=5,mode=Timer.PERIODIC, callback=lambda t:display_change(get_temp(), color))
while True:
sleep(0.1)
main()