硬件部分:先來看硬件連接圖,此次實驗選擇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)的值也就越小
}
三是主函數部分,主函數直接調用光敏傳感器的初始化函數,並在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