单片机毕业设计|基于stm32智能快递箱设计

作者主页:编程千纸鹤

作者简介:Java、前端、Python开发多年,做过高程,项目经理,架构师

主要内容:Java项目开发、Python项目开发、大学数据和AI项目开发、单片机项目设计、面试技术整理、最新技术分享

收藏点赞不迷路  关注作者有好处

文末获得源码

 

项目编号:BS-DPJ-005

一,环境介绍

语言环境:C语言

开发技术:51单片机

二,项目简介

随在当今的社会,网上购物以及线下获取快递己经成为日常生活中很重要的一个组成部分,电子商务的发展也带来了快递业的繁荣。这对快递业而言,是一个巨大的发展机遇,同时也是一个不可忽视的挑战。当前,快件运输的安全性越来越受到大家的重视,对快件的服务要求也越来越高。但就目前的快递行业来说,也面临着这样那样的问题,比较经常遇到送快递的到了,业务不在家,取快递时间对不上等。在此基础上,提出了一种以STM32为核心的智能化快递柜。本快递柜的主要功能有,指纹解锁功能,按键功能,警报功能,继电器柜门开锁功能,验证码功能,主要设计加入了指纹解锁功能。本系统以STM32F103为主控芯片,配置了指纹传感、4*4矩阵键盘、报警提示以及继电器模块等一系列模块,可以使快递员对快递进行安全的存储,这样不仅方便了快递员,也方便了用户,在实现了安全便利地存取快递的同时,也提高了快递行业的服务水平。

三,系统展示

在基于STM32实现的智能快递箱设计中,根据需求分析主要将其分成了以下几个电路:电源电路,LCD1602显示电路,功能按键电路,指示电路,报警电路以及指纹模块电路。

在进行制造之前,要先对每一种电路的工作原理和作用进行理解,然后才能确定电路中可以采用的元件。比如,开关电源回路的电容器具有滤波回路的作用,同时我们也可以根据线路上电流的高低来选择合适的电容器,经过 LED指示回路的电流具有的是限流的作用,一般情况下,可选择的电流范围为100Ω—1 kΩ,只有这样,我们才能很好地把握元件的选用。

在本系列的设计过程中,也遇到了许多的问题,其中就有电路板的排版问题,因为走线的设计不当,最终导致了显示器与键盘的单片机无法协调一致,只能重新设计,浪费了大量的时间。这些都是可以通过自己的努力来降低的。产出焊见。

在将实体制造出来之后,要先用万用表对那些被腐蚀得太厉害并且被焊锡连接起来的部分进行测量,以确保它们之间的连通,然后对电解电容器、二极管、三极管、蜂鸣器等有正负极的部分进行焊接,看它们有没有接反。在确定了正确之后,为了检测一下线路是否可以正常工作,由于感应器的价钱比较贵,所以就没有接上,最后试验了一下线路的情况,发现线路没有问题,屏幕也是亮的。将传感器按每个针脚的次序连在一起,打开后见图

显示器显示为待机状态,按下A键,可进入指纹识别与口令识别的选项,如图

按下B键,进入指纹模式,录入指纹,按D键确认继电器完成开锁显示,如图

按下C键进入密码模式,显示密码,按D键确认继电器完成开锁显示,如图

四,核心代码展示

#include "sys.h" 


//设置向量表偏移地址
//NVIC_VectTab:基址
//Offset:偏移量			 
void MY_NVIC_SetVectorTable(u32 NVIC_VectTab, u32 Offset)	 
{ 	   	 
	SCB->VTOR = NVIC_VectTab|(Offset & (u32)0x1FFFFF80);//设置NVIC的向量表偏移寄存器
	//用于标识向量表是在CODE区还是在RAM区
}
//设置NVIC分组
//NVIC_Group:NVIC分组 0~4 总共5组 		   
void MY_NVIC_PriorityGroupConfig(u8 NVIC_Group)	 
{ 
	u32 temp,temp1;	  
	temp1=(~NVIC_Group)&0x07;//取后三位
	temp1<<=8;
	temp=SCB->AIRCR;  //读取先前的设置
	temp&=0X0000F8FF; //清空先前分组
	temp|=0X05FA0000; //写入钥匙
	temp|=temp1;	   
	SCB->AIRCR=temp;  //设置分组	    	  				   
}
//设置NVIC 
//NVIC_PreemptionPriority:抢占优先级
//NVIC_SubPriority       :响应优先级
//NVIC_Channel           :中断编号
//NVIC_Group             :中断分组 0~4
//注意优先级不能超过设定的组的范围!否则会有意想不到的错误
//组划分:
//组0:0位抢占优先级,4位响应优先级
//组1:1位抢占优先级,3位响应优先级
//组2:2位抢占优先级,2位响应优先级
//组3:3位抢占优先级,1位响应优先级
//组4:4位抢占优先级,0位响应优先级
//NVIC_SubPriority和NVIC_PreemptionPriority的原则是,数值越小,越优先	   
void MY_NVIC_Init(u8 NVIC_PreemptionPriority,u8 NVIC_SubPriority,u8 NVIC_Channel,u8 NVIC_Group)	 
{ 
	u32 temp;	
	MY_NVIC_PriorityGroupConfig(NVIC_Group);//设置分组
	temp=NVIC_PreemptionPriority<<(4-NVIC_Group);	  
	temp|=NVIC_SubPriority&(0x0f>>NVIC_Group);
	temp&=0xf;								//取低四位  
	NVIC->ISER[NVIC_Channel/32]|=(1<<NVIC_Channel%32);//使能中断位(要清除的话,相反操作就OK) 
	NVIC->IP[NVIC_Channel]|=temp<<4;		//设置响应优先级和抢断优先级   	    	  				   
} 
//外部中断配置函数
//只针对GPIOA~G;不包括PVD,RTC和USB唤醒这三个
//参数:
//GPIOx:0~6,代表GPIOA~G
//BITx:需要使能的位;
//TRIM:触发模式,1,下升沿;2,上降沿;3,任意电平触发
//该函数一次只能配置1个IO口,多个IO口,需多次调用
//该函数会自动开启对应中断,以及屏蔽线   	    
void Ex_NVIC_Config(u8 GPIOx,u8 BITx,u8 TRIM) 
{
	u8 EXTADDR;
	u8 EXTOFFSET;
	EXTADDR=BITx/4;//得到中断寄存器组的编号
	EXTOFFSET=(BITx%4)*4; 
	RCC->APB2ENR|=0x01;//使能io复用时钟			 
	AFIO->EXTICR[EXTADDR]&=~(0x000F<<EXTOFFSET);//清除原来设置!!!
	AFIO->EXTICR[EXTADDR]|=GPIOx<<EXTOFFSET;//EXTI.BITx映射到GPIOx.BITx 
	//自动设置
	EXTI->IMR|=1<<BITx;//  开启line BITx上的中断
	//EXTI->EMR|=1<<BITx;//不屏蔽line BITx上的事件 (如果不屏蔽这句,在硬件上是可以的,但是在软件仿真的时候无法进入中断!)
 	if(TRIM&0x01)EXTI->FTSR|=1<<BITx;//line BITx上事件下降沿触发
	if(TRIM&0x02)EXTI->RTSR|=1<<BITx;//line BITx上事件上升降沿触发
} 	  
//不能在这里执行所有外设复位!否则至少引起串口不工作.		    
//把所有时钟寄存器复位		  
void MYRCC_DeInit(void)
{	
 	RCC->APB1RSTR = 0x00000000;//复位结束			 
	RCC->APB2RSTR = 0x00000000; 
	  
  	RCC->AHBENR = 0x00000014;  //睡眠模式闪存和SRAM时钟使能.其他关闭.	  
  	RCC->APB2ENR = 0x00000000; //外设时钟关闭.			   
  	RCC->APB1ENR = 0x00000000;   
	RCC->CR |= 0x00000001;     //使能内部高速时钟HSION	 															 
	RCC->CFGR &= 0xF8FF0000;   //复位SW[1:0],HPRE[3:0],PPRE1[2:0],PPRE2[2:0],ADCPRE[1:0],MCO[2:0]					 
	RCC->CR &= 0xFEF6FFFF;     //复位HSEON,CSSON,PLLON
	RCC->CR &= 0xFFFBFFFF;     //复位HSEBYP	   	  
	RCC->CFGR &= 0xFF80FFFF;   //复位PLLSRC, PLLXTPRE, PLLMUL[3:0] and USBPRE 
	RCC->CIR = 0x00000000;     //关闭所有中断		 
	//配置向量表				  
#ifdef  VECT_TAB_RAM
	MY_NVIC_SetVectorTable(0x20000000, 0x0);
#else   
	MY_NVIC_SetVectorTable(0x08000000,0x0);
#endif
}
//THUMB指令不支持汇编内联
//采用如下方法实现执行汇编指令WFI  
void WFI_SET(void)
{
	__ASM volatile("wfi");		  
}
//关闭所有中断
void INTX_DISABLE(void)
{		  
	__ASM volatile("cpsid i");
}
//开启所有中断
void INTX_ENABLE(void)
{
	__ASM volatile("cpsie i");		  
}
//设置栈顶地址
//addr:栈顶地址
__asm void MSR_MSP(u32 addr) 
{
    MSR MSP, r0 			//set Main Stack value
    BX r14
}

//进入待机模式	  
void Sys_Standby(void)
{
	SCB->SCR|=1<<2;//使能SLEEPDEEP位 (SYS->CTRL)	   
  	RCC->APB1ENR|=1<<28;     //使能电源时钟	    
 	PWR->CSR|=1<<8;          //设置WKUP用于唤醒
	PWR->CR|=1<<2;           //清除Wake-up 标志
	PWR->CR|=1<<1;           //PDDS置位		  
	WFI_SET();				 //执行WFI指令		 
}	     
//系统软复位   
void Sys_Soft_Reset(void)
{   
	SCB->AIRCR =0X05FA0000|(u32)0x04;	  
} 		 
//JTAG模式设置,用于设置JTAG的模式
//mode:jtag,swd模式设置;00,全使能;01,使能SWD;10,全关闭;	   
//#define JTAG_SWD_DISABLE   0X02
//#define SWD_ENABLE         0X01
//#define JTAG_SWD_ENABLE    0X00		  
void JTAG_Set(u8 mode)
{
	u32 temp;
	temp=mode;
	temp<<=25;
	RCC->APB2ENR|=1<<0;     //开启辅助时钟	   
	AFIO->MAPR&=0XF8FFFFFF; //清除MAPR的[26:24]
	AFIO->MAPR|=temp;       //设置jtag模式
} 
//系统时钟初始化函数
//pll:选择的倍频数,从2开始,最大值为16		 
void Stm32_Clock_Init(u8 PLL)
{
	unsigned char temp=0;   
	MYRCC_DeInit();		  //复位并配置向量表
 	RCC->CR|=0x00010000;  //外部高速时钟使能HSEON
	while(!(RCC->CR>>17));//等待外部时钟就绪
	RCC->CFGR=0X00000400; //APB1=DIV2;APB2=DIV1;AHB=DIV1;
	PLL-=2;				  //抵消2个单位(因为是从2开始的,设置0就是2)
	RCC->CFGR|=PLL<<18;   //设置PLL值 2~16
	RCC->CFGR|=1<<16;	  //PLLSRC ON 
	FLASH->ACR|=0x32;	  //FLASH 2个延时周期
	RCC->CR|=0x01000000;  //PLLON
	while(!(RCC->CR>>25));//等待PLL锁定
	RCC->CFGR|=0x00000002;//PLL作为系统时钟	 
	while(temp!=0x02)     //等待PLL作为系统时钟设置成功
	{   
		temp=RCC->CFGR>>2;
		temp&=0x03;
	}    
}		    

void GPIO_Config(GPIO_TypeDef* GPIOx, GPIO_Pin_TypeDef GPIO_pin,uint8_t GPIO_mode)
{
	uint32_t reg_tmp = 0x00; //寄存器缓存
    uint32_t reg_mode = 0x4; //寄存器模式配置   
    int8_t bitPos = -1;//引脚位置
    
	
    //PB3,PB4,PA12
    if((GPIO_pin==GPIO_Pin_12 && GPIOx==GPIOA)||((GPIO_pin &(GPIO_Pin_3|GPIO_Pin_4))!=0 && GPIOx==GPIOB))
    {                                                                                                                                                                                                          
		RCC->APB2ENR|= 0x00000001;//AFIOEN = 1;辅助功能IO时钟使能
		AFIO->MAPR 	|= 0x02000000;//配置AFIO_MAPR,的SWJ_CFG[2:0]位为010:关闭JTAG-DP,启用SW-DP
	}
	//先打开时钟
	     if(GPIOx == GPIOA)RCC->APB2ENR|=1<<2;
	else if(GPIOx == GPIOB)RCC->APB2ENR|=1<<3;	
	else if(GPIOx == GPIOC)RCC->APB2ENR|=1<<4;
	else return;

    //取得引脚位置
    while(GPIO_pin != ((uint16_t)0x0000))
    {
        GPIO_pin >>= 1;
        bitPos++;
    }
    
    //取寄存器配置模式
    reg_mode = GPIO_mode & ((uint32_t)0xF);
    if(GPIO_mode==GPIO_Mode_IPD)GPIOx->BRR = (((uint32_t)0x01) << bitPos);
    else if(GPIO_mode==GPIO_Mode_IPU)GPIOx->BSRR = (((uint32_t)0x01) << bitPos);
    //输出速度配置
    if (((GPIO_mode) & ((uint8_t)0x10)) != 0x00)
    {
        if((GPIO_mode&((uint8_t)0x03))==0x00)reg_mode |= GPIO_Speed_50MHz;//缺省位50MHz速度
    }
    
 
    
    //送到寄存器
    if(bitPos < 8) //引脚GPIO_Pin_x为低8位
    {
        reg_tmp = GPIOx->CRL;//拷贝寄存器
        reg_tmp &= (~(((u32)0xF)<<(bitPos*4)));//置零配置位
        reg_tmp |= ((reg_mode)<<(bitPos*4));//模式配置
        GPIOx->CRL = reg_tmp;
    }
    else//高8位
    {
        reg_tmp = GPIOx->CRH;
        reg_tmp &= (~(((u32)0xF)<<((bitPos-8)*4)));
        reg_tmp |= ((reg_mode)<<((bitPos-8)*4));
        GPIOx->CRH = reg_tmp;
        
    }
}


uint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{
  uint8_t bitstatus = 0x00;
   
  if ((GPIOx->IDR & GPIO_Pin) != (uint32_t)Bit_RESET)
  {
    bitstatus = (uint8_t)Bit_SET;
  }
  else
  {
    bitstatus = (uint8_t)Bit_RESET;
  }
  return bitstatus;
}

void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{
  
  GPIOx->BSRR = GPIO_Pin;
}
void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{ 
  GPIOx->BRR = GPIO_Pin;
}










五,相关作品展示

基于Ja基于微信小程序和安卓APP应用开发的相关作品

基于51单片机等嵌入式物联网开发应用

基于各类算法实现的AI智能应用

基于大数据实现的各类数据管理和推荐系统

 

 

猜你喜欢

转载自blog.csdn.net/BS009/article/details/133303268