基于MSP432P401R送药小车【2021年电赛F题】

news2025/2/3 11:54:57

文章目录

  • 一、任务清单
    • 1. 硬件部分
    • 2. 软件部分
  • 二、神经网络训练
    • 1. 创建数据集
    • 2. 数据采集
    • 3. 数字训练
  • 三、OpenMV数字及其坐标识别
  • 四、巡线
    • 1. 直行
    • 2. 转向
    • 3. 停止
  • 五、路口判断与原路径返回
  • 六、技术交流


由于前边已经用MSP430做过一遍该赛题了,这里就不再重复叙述赛题内容。
也是应各网友需求,用该MSP432主控板再次完成该任务。

Tips: 本篇与上次用MSP430的做法大不一样,其中视觉上,上次采用的模板匹配,本次采用的神经网络数字训练,其各有优势,这里不便讨论,其次在巡线逻辑上,上次采用的是7路灰度,本次采用的12路灰度,逻辑性更强,且在转向时不再使用延迟转弯,因为延迟受电压影响较大,具体实现请见下方描述,然后在药物的装载上,上次采用红外对管检测物品的装卸,本次使用碰撞器模拟,更为方便。

一、任务清单

1. 硬件部分

  • 巡线: 12路灰度传感器
  • 数字识别: OpenMV H7 Plus
  • 数据显示: OLED
  • 药物: 碰撞器(模拟)
  • 稳压: LM2596S DC-DC
  • 视觉云台: 3D建模打印
  • 主控板: MSP432P401R
  • 电机: TT霍尔编码电机
  • 驱动模块: TB6612
  • 电源: 18650锂电池
  • 仪器: 3D打印机
  • 其他: 万向轮、开关、杜邦线等。

在这里插入图片描述

2. 软件部分

  • 训练网址: EDGE IMPULSE
  • 编译器: Keil、OpenMV IDE
  • 建模软件: SketchUp Pro 2022
  • 切片软件: Cura
  • 编程方式: 库函数
  • 编程语言: C、Python

二、神经网络训练

1. 创建数据集

打开OpenMV IDE,在菜单栏选择 “工具” —> “数据集编辑器” —> “新数据集”,创建好数据集后在左侧的菜单项中选择新建文件夹用于存储每个标签下的照片,即新建八个文件夹,依次在每个文件夹中采集照片。
在这里插入图片描述
在这里插入图片描述

2. 数据采集

依次在每个文件夹中采集对应图片
在这里插入图片描述

3. 数字训练

将拍出来的数据放置EDGE IMPULSE训练,这样训练出来的效果远比用手机拍出来的效果好很多哦。注意在拍照过程中尽量让背景更简单单纯一点,不然训练后的结果可能部分数字会受背景的影响。
在这里插入图片描述
Tips: 每张照片建议在100张左右,对易混淆的数字加量采集,如数字5和数字6可以加大到150张左右,且在采集过程中多选择不同方位进行拍照,在标签画框时只用框出中间数字即可,四周有加黑色框的尽量不要将框也标记进去,因为这样会影响训练效果。

三、OpenMV数字及其坐标识别

视觉这块就比较简单易行了,将训练好的模型文件导入OpenMV中打开,即可准确识别数字及相应坐标,所以操作空间很小,只需要根据单片机发过来的指令完成指定动作再将数据返回到单片机即可,总共分为4次识别,第一次是在开始装药品时识别目的病房,第二次是在路口二处进行识别,第三次是在路口三,第四次是在路口四或路口五,所以最多也只需要进行四次识别操作,相对是非常简单的。

那怎么判断是否识别准确呢?可以通过OpenMV板载的LED灯进行闪烁提示,例 若识别到数字2则LED灯闪烁俩次;或者也可以通过发送到单片机的数据进行判断,将其显示在OLED显示屏上。

具体如下:

while(True):
    clock.tick()

    img = sensor.snapshot()

    if uart.any():
        data=uart.readline().decode().strip()  #转化为字符串接收
    if data=='8':
        first=0
        second=0
        three=0
        four=0
        second_flag=1
        three_flag=1
        four_flag=1
        x=0
        flag=0
        data='0'


    if first==0:
        first=get_first_num(img)
        if first:
            second_flag=0
            print(first)
            send_num(first)
        #led_show(first)

    if second_flag==0:
        if data=='1':
            print('ok1')
            second=get_num(img)
            if second==0:
                second_flag=1
                three_flag=0
                data='0'
                print("go")
                uart.write("F")
            elif second<=100:
                second_flag=1
                three_flag=0
                data='0'
                print(second)
                uart.write("L")
            elif second>100:
                second_flag=1
                three_flag=0
                data='0'
                print(second)
                uart.write("R")

    if three_flag==0:
        if data=='2':
            print('ok2')
            three=get_num(img)
            if three<=100:
                three_flag=1
                four_flag=0
                data='0'
                print(three)
                uart.write("l")
            elif three>100:
                three_flag=1
                four_flag=0
                data='0'
                print(three)
                uart.write("r")

    if four_flag==0:
        if data=='3':
            print('ok3')
            four=get_num(img)
            if four<=100:
                four_flag=1
                data='0'
                print(four)
                uart.write("a")
            elif four>100:
                four_flag=1
                data='0'
                print(four)
                uart.write("b")


四、巡线

1. 直行

这里我采用的是12路灰度传感器,其中中间6个探头用来完成直行动作,也就是该6个探头只用作红白色处理。

具体如下:

void XunXian(void)
{			
	if( l4==0 && l5==0 && l6==1 && l7==1 && l8==0 && l9==0 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);			
	}
	//左边
	else if( l4==0 && l5==1 && l6==1 && l7==0 && l8==0 && l9==0 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 350);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);			
	}		
	else if( l4==1 && l5==1 && l6==0 && l7==0 && l8==0 && l9==0 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 250);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);					
	}
	else if( l4==1 && l5==0 && l6==0 && l7==0 && l8==0 && l9==0 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 100);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);			
	}
	
	//右边
	else if( l4==0 && l5==0 && l6==0 && l7==1 && l8==1 && l9==0 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 330);					
	}
	else if( l4==0 && l5==0 && l6==0 && l7==0 && l8==1 && l9==1 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 230);			
	}
	else if( l4==0 && l5==0 && l6==0 && l7==0 && l8==0 && l9==1 )
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 100);				
	}
}

2. 转向

本次任务中的转向均为循环转向,也就是在执行转向操作时会一直向着某个方向转,直至某个探头的状态发生变化才退出循环继续执行巡线操作,这样就不再因为延迟转向受电压影响了。

具体如下:

	//左转
	if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==4)
	{
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	
	//右转
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==5)
	{
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}	
	}	

3. 停止

周边4个探头用来执行停止动作,也就是该4个探头只用作黑白色处理(不受红色影响)。当其中某一个探头状态发生变化时,其执行停止指令,该指令也只会在停止点黑色虚线处被执行。

具体如下:

if((l1==1||l2==1||l11==1||l12==1) &&(over==1))
{
	Stop();
	MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
	MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
}	

五、路口判断与原路径返回

该赛图总共分为5个交叉路口,则我将该5个交叉路口封装为5个函数,每当要行驶到某个路口时执行相应的路口操作函数,例如路口一,该路口只对病房①和病房②有效,即可根据OpenMV识别返回到的数字可判断,如果是数字①和数字②,则执行路口一的执行函数,否则直接跳过路口一,直接执行路口二的执行函数,再根据路口二识别到的数字进行判断,如果有相应的数字则根据OpenMV返回的数字坐标进行左右转向判断,否则直接进入路口三再次进行数字识别,根据识别到的数字坐标判断左右转向,如果得到的是左转指令,则进入路口四做最后一次识别,否则进入路口五做最后一次是识别,根据最后一次识别到的数字坐标进入到最后的病房,随机按照原路径返回到药房。

路口于下图所示:
在这里插入图片描述
其中,路口一、二、三通过中间6个探头进行判断,当最中间的四个探头状态都发生变化是,则判定此时正处于路口处,路口四和路口五则需在中间6个探头的基础上外加旁边的俩个探头进行判断,因为是“T”型路口,若最左、右俩边的三个探头状态发生改变而对应边的状态并没有发生改变则判定此时正处于“T”型路口处。

原路径返回则是通过在每进行一次转向时会做一次记录,直至记录到成功将药品送到病房后,再依次按照记录的值进行路口转向,直至最外围4个探头状态发生变化,则判定成功返回到药房。

具体操作如下:

void Lukou1(void)
{
	
	if(openmv_rxflag==1)
	{		
			//去往1号病房的十字路口左转
		if( (l5==1&&l6==1&&l7==1&&l8==1)&& (room1_flag==0))
		{
			room1_flag=1;
			delay_ms(110);
			Left();
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
			delay_ms(300);
			while(1)
			{
				l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
				if(l7==1) break;
			}
			stop_flag=1;
		}
		//1号病房回到药房的十字路口右转
		else if( (l5==1&&l6==1&&l7==1&&l8==1) && (room1_flag==2))
		{
			room1_flag=3;
			delay_ms(100);
			Right();
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
			delay_ms(350);
			while(1)
			{
				l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
				if(l6==1) break;
			}
			over=1;
		}
	}
	
	if(openmv_rxflag==2)
	{
		//去往2号病房的十字路口右转
		if( (l5==1&&l6==1&&l7==1&&l8==1) && (room2_flag==0))
		{
			room2_flag=1;
			delay_ms(100);		
			Right();
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
			delay_ms(350);
			while(1)
			{
				l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
				if(l6==1) break;
			}
			stop_flag=1;
		}
		//2号病房回到药房的十字路口左转
		else if( (l5==1&&l6==1&&l7==1&&l8==1) && (room2_flag==2))
		{
			room2_flag=3;
			delay_ms(110);
			Left();
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
			MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
			delay_ms(300);
			while(1)
			{
				l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
				if(l7==1) break;
			}
			over=1;
		}		
	}
	
	//1、2号病房停在病房门口
	if((l1==1||l2==1||l11==1||l12==1) && stop_flag==1)
	{		
		stop_flag=2;
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		room1_flag=2;
		room2_flag=2;
		button_flag=1;
		while(button_flag)
		{
			button_flag=gpio_get(GPIO_PORT_P7, GPIO_PIN7);
		}
		Back();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}		
	}

	//返回到药房门口
	if((l1==1||l2==1||l11==1||l12==1) &&(over==1))
	{
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		TimerA1_Disable();
		OLED_ShowString(30, 4, (uint8_t *)"T=    S", 16);
		OLED_ShowNum(47,4,Timer,2,16);
	}		
}


void Lukou2(void)
{
	//接收到openmv发过来的左转标志
	if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==1)
	{
		LR_flag=11;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
		stop_flag=1;
		room3_flag=1;
	}
	//接收到openmv发过来的右转标志
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==2)
	{
		LR_flag=22;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
		stop_flag=1;
		room4_flag=1;
	}
	//接收到openmv发过来的直行标志
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==3)
	{
		LR_flag=33;
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);		
	}
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && (room3_flag==1))
	{
		room3_flag=2;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
	}
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && (room4_flag==1))
	{
		room4_flag=2;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && (room3_flag==2||room4_flag==2))
	{
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);	
		over=1;
	}

	//停在病房门口
	if( (l1==1||l2==1||l11==1||l12==1) && stop_flag==1)
	{		
		stop_flag=2;
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		button_flag=1;		
		while(button_flag)
		{
			button_flag=gpio_get(GPIO_PORT_P7, GPIO_PIN7);
		}
		Back();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}		
	}
	//返回到药房门口
	else if((l1==1||l2==1||l11==1||l12==1) &&(over==1))
	{
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		TimerA1_Disable();
		OLED_ShowString(30, 4, (uint8_t *)"T=    S", 16);
		OLED_ShowNum(47,4,Timer,2,16);
	}	
}

void Lukou3(void)
{
	//接收到openmv发过来的左转标志
	if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==4)
	{
		LR_flag=44;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	//接收到openmv发过来的右转标志
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==5)
	{
		LR_flag=55;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}	
	}	
}

void Lukou4(void)
{
	//接收到openmv发过来的左转标志
	if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==7)
	{
		LR_flag=77;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
		stop_flag=1;
	}
	//接收到openmv发过来的右转标志
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==8)
	{
		LR_flag=88;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
		stop_flag=1;
	}
	else if( (l5==0&&l8==1&&l9==1&&l10==1) && LR_flag==77)
	{
		LR_flag=777;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
	}
	else if( (l5==0&&l8==1&&l9==1&&l10==1) && LR_flag==777)
	{
		LR_flag=7777;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
	}
	else if( (l3==1&&l4==1&&l5==1&&l8==0) && LR_flag==88)
	{
		LR_flag=888;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	else if( (l5==0&&l8==1&&l9==1&&l10==1) && LR_flag==888)
	{
		LR_flag=8888;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
	}
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && ((LR_flag==7777)||(LR_flag==8888)))
	{
		over=1;
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);		
	}
	
	//停在病房门口
	if( (l1==1||l2==1||l11==1||l12==1) && stop_flag==1)
	{		
		stop_flag=2;
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		button_flag=1;		
		while(button_flag)
		{
			button_flag=gpio_get(GPIO_PORT_P7, GPIO_PIN7);
		}
		Back();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}			
	}
	//返回到药房门口
	else if((l1==1||l2==1||l11==1||l12==1) &&(over==1))
	{
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		TimerA1_Disable();
		OLED_ShowString(30, 4, (uint8_t *)"T=    S", 16);
		OLED_ShowNum(47,4,Timer,2,16);
	}	
}

void Lukou5(void)
{
	//接收到openmv发过来的左转标志
	if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==7)
	{
		LR_flag=77;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
		stop_flag=1;
	}
	//接收到openmv发过来的右转标志
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && LR_flag==8)
	{
		LR_flag=88;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
		stop_flag=1;
	}
	else if( (l5==0&&l8==1&&l9==1&&l10==1) && LR_flag==77)
	{
		LR_flag=777;
		delay_ms(100);
		Right();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 320);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(350);
		while(1)
		{
			l6=gpio_get(GPIO_PORT_P9, GPIO_PIN3);			
			if(l6==1) break;
		}
	}
	else if( (l5==0&&l8==1&&l9==1&&l10==1) && LR_flag==777)
	{
		LR_flag=7777;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	else if( (l3==1&&l4==1&&l5==1&&l8==0) && LR_flag==88)
	{
		LR_flag=888;
		delay_ms(100);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	else if( (l5==0&&l8==1&&l9==1&&l10==1) && LR_flag==888)
	{
		LR_flag=8888;
		delay_ms(100);	
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);		
		delay_ms(300);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}
	}
	else if( (l5==1&&l6==1&&l7==1&&l8==1) && ((LR_flag==7777)||(LR_flag==8888)))
	{
		over=1;
		Forward();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 400);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 380);		
	}
	
	//停在病房门口
	if( (l1==1||l2==1||l11==1||l12==1) && stop_flag==1)
	{		
		stop_flag=2;
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		button_flag=1;		
		while(button_flag)
		{
			button_flag=gpio_get(GPIO_PORT_P7, GPIO_PIN7);
		}
		Back();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		Left();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 300);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 280);	
		delay_ms(500);
		while(1)
		{
			l7=gpio_get(GPIO_PORT_P6, GPIO_PIN3);			
			if(l7==1) break;
		}			
	}
	//返回到药房门口
	else if((l1==1||l2==1||l11==1||l12==1) &&(over==1))
	{
		Stop();
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_1, 0);
		MAP_Timer_A_setCompareValue(TIMER_A0_BASE, TIMER_A_CAPTURECOMPARE_REGISTER_2, 0);	
		TimerA1_Disable();
		OLED_ShowString(30, 4, (uint8_t *)"T=    S", 16);
		OLED_ShowNum(47,4,Timer,2,16);
	}
}


六、技术交流

疑难解答或技术交流联系下方wx即可👇👇👇

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

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

相关文章

Java培训:什么是Busy spin?为什么要使用Busy spin?

Busy spin(繁忙自旋)是一种线程等待的技术&#xff0c;它通过循环检查条件来等待某个事件或条件的发生&#xff0c;而不进行阻塞或休眠。 通常情况下&#xff0c;线程等待事件发生的方式是使用阻塞或休眠操作&#xff0c;这样线程会释放CPU资源&#xff0c;其他线程可以继续执行…

Qt6 Qt Quick UI原型学习QML第二篇

Qt6 Qt Quick UI原型学习QML第二篇 界面效果QML语法语法讲解核心要素项目元素矩形元素文本元素图像元素MouseArea元素 界面效果 QML语法 import QtQuick 2.12 import QtQuick.Window 2.12Window {id: rootvisible: truewidth: 640height: 480title: qsTr("QML学习第二篇&…

【题解】 模拟赛2 题解

T1 假设商品价格为x 618:int(x*0.66) 211:x-(x/100)*35 两者比较一下大小即可 #include<bits/stdc.h> using namespace std;int x,x1,x2;int main(){scanf("%d",&x);x1 x*0.66;x2 x-(x/100)*35;if (x1 x2) printf("both\n%d",x1);if (x1 &g…

浏览器打开PDF标题乱码

问题 使用 itext5 用pdf模板生成预览pdf乱码问题 解决办法 使用pdf编辑器打开之后&#xff0c;选择 文件>> 属性&#xff0c; 修改乱码的标题。

【业务功能篇45】SSM整合shiro项目:web.xml执行顺序

web.xml 的加载顺序是&#xff1a;ServletContext -> context-param -> listener -> filter -> servlet 学习shiro时&#xff0c;需要配置shiro &#xff0c;我们需要在filter过滤器之前&#xff0c;先初始化好shiro组件&#xff0c;不然请求认证无法走到shiro,根据…

plt.text()函数解析

plt.text(x, y, s, fontsize, verticalalignment,horizontalalignment,rotation , *kwargs) 参数&#xff1a; x,y:表示坐标值上的值s:表示说明文字fontsize:表示字体大小verticalalignment&#xff1a;垂直对齐方式 &#xff0c;参数&#xff1a;[ ‘center’ | ‘top’ | ‘…

【公益】Q学友联合东湖街道开展“星级大厨来做客”技能培训活动

“大家一定要用温水和面&#xff0c;和面时要注意方向和力度&#xff0c;往同一个方向揉面……”在东湖街道综合文体服务中心一楼的中式面点培训现场&#xff0c;飘荡着阵阵面香&#xff0c;充斥着欢声笑语。 为进一步丰富居民业余文化生活&#xff0c;提高灵活就业人员的职业技…

手把手教你搭建SpringCloud项目:什么是微服务?一看就会系列!

什么是微服务&#xff1f;一看就会系列&#xff01; 一、手把手教你搭建SpringCloud项目&#xff08;一&#xff09;图文详解&#xff0c;傻瓜式操作 二、手把手教你搭建SpringCloud项目&#xff08;二&#xff09;生产者与消费者 三、手把手教你搭建SpringCloud项目&#x…

mpVue 微信小程序基于vant-weapp 组件的二次封装TForm 表单组件(修改源码插槽使用)

一、前言 1、mpVue微信小程序不支持动态组件&#xff08;<component> &#xff09; 2、mpVue微信小程序不支持动态属性及事件穿透&#xff08;$attrs和$listeners&#xff09; 3、mpVue微信小程序不支持render函数 二、最终效果 三、配置参数&#xff08;Attributes&…

Qt6 Qt Quick UI原型学习QML第三篇

文章目录 效果QML代码ClickableImage.qml文件Image&#xff08;图片&#xff09;元素 解释 MyQML.qml文件 解释&#xff1a;Window元素、Item元素、Image元素、MouseArea元素、Column元素、Row元素、Grid元素、Flow元、Repeater元素 效果 QML代码 ClickableImage.qml文件 图像…

[JavaScript] 第三章 Chrome 浏览器中执行 JavaScript

系列文章目录 [JavaScript] 第一章 暂无 [JavaScript] 第一章 暂无 [JavaScript] 第三章 Chrome 浏览器中执行 JavaScript 文章目录 系列文章目录前言1、准备工作1.1、创建html工程1.2、创建html文件夹&#xff0c;存放html文件1.3、创建JavaScript演示html1.4、通过idea打开页…

Unity打包窗口化放大、缩小、拖拽功能、无边框设置 C#

Unity打包Windows窗口实现放大、缩小、拖拽、无边框 文章目录 Unity打包Windows窗口实现放大、缩小、拖拽、无边框前言一、引入 user32.dll二、使用步骤1.引入库2.功能封装3.效果图如下&#xff0c;绑定自定义按钮 总结 前言 Unity无边框设置、窗口化放大、缩小、拖拽 提示&am…

蓝牙耳机选购攻略:开放式耳机篇!如何选购开放式耳机?开放式蓝牙耳机哪些品牌比较好?过来人告诉你如何选购开放式耳机!

作为一个耳机爱好者&#xff0c;最近更是喜欢上了开放式蓝牙耳机&#xff0c;实际用过的起码有十几款&#xff0c;但其实最终能留下来的也只有四五款。由于前期并不知道应该如何选择开放式耳机&#xff0c;经常都会高价买到些质量差、音质也不好、漏音大的开放式耳机&#xff0…

DKN和KGC阅读

1. DKN 作者将外部知识图包含的知识融入新闻嵌入。 &#xff08;1&#xff09;将新闻标题单词词嵌入&#xff0c;单词链接的实体嵌入&#xff0c;以及实体的上下文嵌入(邻居实体嵌入的平均)建模为CNN输入的三个通道。 &#xff08;2&#xff09;然后使用KCNN模型&#xff0c…

排序【数据结构】

1、排序的概念 排序&#xff1a; 所谓排序&#xff0c;就是使一串记录&#xff0c;按照其中的某个或某些关键字的大小&#xff0c;递增或递减的排列起来的操作。稳定性&#xff1a; 假定在待排序的记录序列中&#xff0c;存在多个具有相同的关键字的记录&#xff0c;若经过排序…

一个月学通Python(十九):Cookie和Session是什么(Web开发)

专栏介绍 结合自身经验和内部资料总结的Python教程&#xff0c;每天3章&#xff0c;1个月就能全方位的完成Python的学习并进行实战开发&#xff0c;学完了定能成为大佬&#xff01;加油吧&#xff01;卷起来&#xff01; 全部文章请访问专栏&#xff1a;《Python全栈教程&…

C++:这门语言优势在哪?命名空间以及缺省参数?

文章目录 C的优势解决命名空间的问题 缺省参数 C的优势 C和C语言比起来有许多优势&#xff0c;这里我们先举一个例子&#xff0c;后续进行补充 解决命名空间的问题 首先看这样的代码&#xff1a; #include <stdlib.h> #include <stdio.h>int rand 0;int main(…

k8s服务发现之第二弹Service详解

创建 Service Kubernetes Servies 是一个 RESTFul 接口对象&#xff0c;可通过 yaml 文件创建。 例如&#xff0c;假设您有一组 Pod&#xff1a; 每个 Pod 都监听 9376 TCP 端口每个 Pod 都有标签 appMyApp apiVersion: v1 kind: Service metadata:name: my-service spec:s…

创建users子应用

1.创建&#xff1a;(venv) xxxx>python manage.py startapp users 2.移动到apps里面&#xff1b; 3.在settings里面注册子应用&#xff1b; # 注册子应用 apps/users apps/users/apps name apps.users 这个路径是什么就写什么 INSTALLED_APPS [ # 用[../namage.py st…

架构训练营学习笔记:4-3存储架构模式之分片架构和分区架构

分片架构的本质&#xff1a; 分片架构能提升写性能和存储性能&#xff0c;对应的主备架构的本质是备份&#xff0c;主从架构本质提升读性能。 分片架构的两个关键点&#xff1a;分片规则跟路由规则 分片规则&#xff1a; 选择基数比较大的某个数据键值&#xff0c;让数据均匀…