STM32學習之搭建光敏二極管電路並採集判斷光強

硬件部分:先來看硬件連接圖,此次實驗選擇ADC3的通道7,硬件原理圖如圖1所示,光敏電阻的原理在圖1中已經說明,這裏就不再多說。圖2是stm32的部分引腳圖。


圖1


圖 3

軟件部分:

軟件部分主要是三個方面,一是使用ADC時對ADC的初始化,初始化之後獲取某個ADC某個通道的值,這裏就是ADC3的通道7,而是哪一個通道使通過函數u16 Get_Adc3(u8 ch)  中的ch傳入的。主要代碼如下:

//初始化ADC3
//這裏我們僅以規則通道爲例
//我們默認僅開啓通道7   
void  Adc3_Init(void)
{      
ADC_InitTypeDef ADC_InitStructure; 


RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC3,ENABLE);  //使能ADC3通道時鐘
RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC3,ENABLE);//ADC復位
RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC3,DISABLE);//復位結束
   

ADC_DeInit(ADC3);  //復位ADC3,將外設 ADC3的全部寄存器重設爲缺省值

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式: 獨立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模數轉換工作在單通道模式
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模數轉換工作在單次轉換模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //轉換由軟件而不是外部觸發啓動
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC數據右對齊
ADC_InitStructure.ADC_NbrOfChannel = 1; //順序進行規則轉換的ADC通道的數目
ADC_Init(ADC3, &ADC_InitStructure); //根據ADC_InitStruct中指定的參數初始化外設ADCx的寄存器  


ADC_Cmd(ADC3, ENABLE); //使能指定的ADC3

ADC_ResetCalibration(ADC3); //使能復位校準  
 
while(ADC_GetResetCalibrationStatus(ADC3)); //等待復位校準結束

ADC_StartCalibration(ADC3); //開啓AD校準
 
while(ADC_GetCalibrationStatus(ADC3)); //等待校準結束
}  
//獲得ADC3某個通道的值
//ch:通道值 0~16
//返回值:轉換結果
u16 Get_Adc3(u8 ch)   
{
  //設置指定ADC的規則組通道,一個序列,採樣時間
ADC_RegularChannelConfig(ADC3, ch, 1, ADC_SampleTime_239Cycles5 ); //ADC3,ADC通道,採樣時間爲239.5週期
ADC_SoftwareStartConvCmd(ADC3, ENABLE); //使能指定的ADC3的軟件轉換啓動功能
while(!ADC_GetFlagStatus(ADC3, ADC_FLAG_EOC ));//等待轉換結束
return ADC_GetConversionValue(ADC3); //返回最近一次ADC3規則組的轉換結果

二是因爲我們的電壓輸入需要用到GPIOF,所以我們需要對其進行初始化:

void Lsens_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF,ENABLE);//使能PORTF時鐘
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;//PF9 anolog輸入
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
  GPIO_Init(GPIOF, &GPIO_InitStructure);
  Adc3_Init();
//這裏我們可以直接調用ADC的初始化,從而將整個函數封裝成光敏傳感器的初始化函數,一會兒主函數中直接調用

}

對光敏傳感器初始化之後需要將我們ADC3採集到的值進行一些轉換,從而直接通過LCD上顯示出來的值的大小判斷光強,其範圍設置爲0~100。封裝函數如下:

//讀取Light Sens的值
//0~100:0,最暗;100,最亮 
u8 Lsens_Get_Val(void)
{
u32 temp_val=0;
u8 t;
for(t=0;t<LSENS_READ_TIMES;t++)
{
temp_val+=Get_Adc3(LSENS_ADC_CHX); //讀取ADC值
delay_ms(5);
}

temp_val/=LSENS_READ_TIMES;//得到平均值 

//3.3V對應4096,而光敏電阻分得的電壓值不可能大於3.3V,也就是說ADC採集到的值不可能大於4096,這裏以最大4000爲界

if(temp_val>4000)temp_val=4000;

//光敏電阻分得的電壓值越大,則(temp_val/40)的值越大,即光敏電阻分得的電壓值越大;也就是是說,光照強度越弱,所以這裏在分成0~100後,用100-100-(temp_val/40)的值也就越小

return (u8)(100-(temp_val/40));

}

三是主函數部分,主函數直接調用光敏傳感器的初始化函數,並在while循環中不停的獲取ADC3採集後的光照強度值,代碼如下:

#include <stdio.h>
#include "stm32f10x.h"
#include "led.h"
#include "delay.h"
#include "key.h"
#include "time.h"
#include "usart.h"
#include "lcd.h"
#include "adc.h"
#include "lsens.h"


int main(void)
{
u8 adcx; 
delay_init();     //延時函數初始化  
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//設置中斷優先級分組爲組2:2位搶佔優先級,2位響應優先級
uart_init(115200); //串口初始化爲115200
LED_Init();   //初始化與LED連接的硬件接口
  LCD_Init(); //初始化LCD
Lsens_Init(); //初始化光敏傳感器
POINT_COLOR=RED;//設置字體爲紅色  
//顯示提示信息      
LCD_ShowString(30,50,200,16,24,"Name:Li Hai");
LCD_ShowString(30,80,200,16,24,"Age:25");
LCD_ShowString(30,110,200,16,24,"Tel:XX");
LCD_ShowString(30,140,200,16,24,"Date:2018/3/24");  
POINT_COLOR=BLUE;//設置字體爲藍色
LCD_ShowString(30,170,200,16,24,"LSENS_VAL:");             
while(1)
{
adcx=Lsens_Get_Val();
LCD_ShowxNum(30+10*12,170,adcx,3,24,0);//顯示ADC的值 
LED2_REV;
delay_ms(250);
}
}

最後將HEX文件燒寫如單片機中,顯示結果如圖3所示。


圖3

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章