01. 光敏传感器简介
9 W/ T8 m8 x3 i% s4 q光敏传感器是最常见的传感器之一,它的种类繁多,主要有:光电管、光电倍增管、光敏电阻、光敏三极管、太阳能电池、红外线传感器、紫外线传感器、光纤式光电传感器、色彩传感器、CCD 和 CMOS 图像传感器等。光传感器是目前产量最多、应用最广的传感器之一,它在自动控制和非电量电测技术中占有非常重要的地位。: y$ {4 |+ @8 f( W6 D3 p
7 [5 {% }1 }2 I3 A/ w% @# |光敏传感器是利用光敏元件将光信号转换为电信号的传感器,它的敏感波长在可见光波长附近,包括红外线波长和紫外线波长。光传感器不只局限于对光的探测,它还可以作为探测元件组成其他传感器,对许多非电量进行检测,只要将这些非电量转换为光信号的变化即可。. b4 `: N6 r* ~0 U2 d
" l& g! m; q4 ?8 z$ d! K8 O
探索者 STM32F4 开发板板载了一个光敏二极管(光敏电阻),作为光敏传感器,它对光的变化非常敏感。光敏二极管也叫光电二极管。光敏二极管与半导体二极管在结构上是类似的,其管芯是一个具有光敏特征的 PN 结,具有单向导电性,因此工作时需加上反向电压。无光照时,有很小的饱和反向漏电流,即暗电流,此时光敏二极管截止。当受到光照时,饱和反向漏电流大大增加,形成光电流,它随入射光强度的变化而变化。当光线照射 PN 结时,可以使 PN 结中产生电子一空穴对,使少数载流子的密度增加。这些载流子在反向电压下漂移,使反向电流增加。因此可以利用光照强弱来改变电路中的电流。
6 S; |0 |) V" k& `( |# a6 C0 F/ g8 F+ C. ~- G# Q3 i) h$ [1 p( O
利用这个电流变化,我们串接一个电阻,就可以转换成电压的变化,从而通过 ADC 读取电压值,判断外部光线的强弱。7 o* Q, N; w0 n3 Z) a
* L" M" \8 _6 ^% ^5 ~8 O02. 硬件模块1 X \4 r+ [4 M* W7 A8 @, r4 _
用到的硬件资源有:" L" l$ p9 s" H0 N+ S, w" a: }" ?! }
1) 指示灯 DS0
: |& a) W- V- K3 y2) TFTLCD 模块( i( o [9 |; Z( h* r( H2 `0 L
3) ADC) J1 F4 i5 B; D
4) 光敏传感器
+ H2 S0 D6 N% i
! y7 Y, Q: ]6 E: K* ^, e, `
$ P4 S6 v0 |7 P4 u d9 [" u
& F, S2 t5 H6 z) z, w7 J
LS1 是光敏二极管(实物在开发板摄像头接口右侧),R58 为其提供反向电压,当环境光线变化时,LS1 两端的电压也会随之改变,从而通过 ADC3_IN5 通道,读取LIGHT_SENSOR(PF7)上面的电压,即可得到环境光线的强弱。光线越强,电压越低,光线越暗,电压越高。
; c! Y, g% f8 B% V# K P7 [( q7 B. i. L0 P9 c _2 X; K
03. 程序设计
; |# `$ f% r7 ?adc.h文件
0 {; I% x4 c- q) ^- //-----------------------------------1 Y B4 P, |+ J7 _0 Q8 p
- //ADC3通道初始化
E U: s5 M `) q) c6 G - void ADC3_Init(void);
' D& k8 l* ?4 O) S' S# P, W" Y - //获取ADC3某个通道的值- k1 o0 i4 D! p! G6 M! V2 Z
- u16 Get_Adc3(u8 ch);1 Q0 B4 y$ |$ `9 T% t b- k
- //获取某个通道给定次数采样的平均值" {8 v9 I9 W4 Y, b+ j% f' {
- u16 Get_Adc3_Average(u8 ch, u8 times);
复制代码
8 z. U6 u! ^3 Q9 l( _' Ladc.c文件* d p. {1 n7 A( Y5 s
/ G1 H- o$ X' a0 r6 l- //-------------------------ADC3------------------------
+ k' M( c" [- k* S" i* F8 p4 z - //ADC3通道初始化
* I7 j5 n. L% C% \) [ - void ADC3_Init(void)
* p+ b& s5 ]! ?( a( T1 Y - {
" l( L: V0 T; t1 g* f% J - GPIO_InitTypeDef GPIO_InitStructure; g- W9 U. D% |1 k" e- ~5 z
- ADC_CommonInitTypeDef ADC_CommonInitStructure;- B" _! T3 B/ {. |8 u
- ADC_InitTypeDef ADC_InitStructure;
/ M7 j2 N/ l% O - : ?% |. U5 T E( Q' P! n
- //开启GPIO时钟 PA5
; U+ W, K! V# z$ o. L {- k - RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF, ENABLE);
8 [" L9 l) g, m3 \; y% Q7 Q8 l; ]7 D - * `) S7 Z+ M9 ~& ^* ~1 L- r
- //开启ADC1时钟
1 N, A5 c7 V+ [. {# d& L" g( X* n - RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC3, ENABLE); 6 K+ }' F# |' V0 @) t9 n* y
- $ z- r2 e" S" w+ W5 x. T. |5 y: j
- //GPIO初始化 初始化为模拟功能
. p% A# `, `) Q& r0 ]- X/ z2 @ - GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
( P9 J' q: D* Z( E! a, t - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
/ R& ?+ P* G+ }. v6 a, j* { - GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
2 l: h2 u% H" D8 h! |7 [+ k - GPIO_Init(GPIOF, &GPIO_InitStructure);
T; }. [/ M b" T3 D, W -
8 T. X, Q4 L1 {! L% Y - RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC3,ENABLE); //ADC1复位; I3 N/ c7 f9 M* I
- RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC3,DISABLE); //复位结束 @" q: l4 f7 P; I. e% j' I
- 7 m- E, \5 e. D$ h
-
# X) i$ J' M* l$ s. O" x - //初始化ADC_CCR寄存器
' r" C" M2 O7 F0 W' \- F - $ } d- x" v8 P$ {
- ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;! n% L; K- p( D W8 X
- ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;1 F# W) {! I7 Y; p: [
- ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;# d& }4 {4 ]. o/ ^( c: Z
- ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;, R0 H) Q5 m( Y7 ]& r
- ADC_CommonInit(&ADC_CommonInitStructure);. z+ l2 P7 x/ x+ P& Q8 k
- # _0 `; P: T3 y4 @7 A
- E0 `) M5 O; p- m# b! T- //初始化ADC16 ~( } ?! [7 N, \) R" J4 l
- ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
2 [# S! u7 `; Y4 r0 E: r - ADC_InitStructure.ADC_ScanConvMode = DISABLE;5 f8 s. C: B$ w4 O7 I
- ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
1 b+ a$ O( e& O F% h - ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;1 f& ?6 H% x' O/ n" Z
- ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;% s$ M/ Q- M- U4 w# m4 P
- ADC_InitStructure.ADC_NbrOfConversion = 1;
( L& Y" G! t- j* n: e$ R2 R# m% E - % Y9 {* K5 P3 Y- _. G
- ADC_Init(ADC3, &ADC_InitStructure);# k6 `; K- I: `/ n* F" ~* R
& ]( O" |8 Y0 w& J3 ~- ^6 h" s! [; a9 |( J7 Q6 w
- //设置ADC规则组通道 一个序列 采样时间
6 E; V' W# ~) o( g }# J - //ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 1, ADC_SampleTime_480Cycles);6 ]: w7 d* ?, h$ s+ J7 t7 V3 u
- & H% S+ G& |1 K) a% b
3 X' q) h" ?/ L, j& h- //使能ADC" N2 e/ S+ I/ n( c
- ADC_Cmd(ADC3, ENABLE);% J5 x% Z- R5 y( W) N
-
# ]0 b1 n0 x" H - }( K9 a- e; x d* w
6 a. e! {+ Q' r7 [9 F- //获取ADC3某个通道的值1 C& \( q1 ?4 l6 c. V
- u16 Get_Adc3(u8 ch)0 p( r0 M! R2 `' k# Z
- {
6 I7 E6 x& u3 |5 } - //设置ADC规则组通道 一个序列 采样时间
& x0 `9 \! G( h4 T" O& a9 [( D - ADC_RegularChannelConfig(ADC3, ch, 1, ADC_SampleTime_480Cycles);
, n# `$ x% k5 p; c -
# q: l/ s1 p! d8 s4 j0 o - //开启ADC3的软件转换启动功能3 C: R2 Z* J; {( j
- ADC_SoftwareStartConv(ADC3);1 |2 ^# a5 k8 D1 o5 f7 [5 _
-
( [: l+ }7 W' ?5 V: \# _7 r! h: N# h0 I - //等待转换结束3 C9 p! U9 d) [3 `! f" ]
- while(ADC_GetFlagStatus(ADC3, ADC_FLAG_EOC) == RESET)
' B' c) _; O; V0 w/ _ - ;
/ ~# m# _/ u" h, D& Q - $ ?6 }- t8 v( d4 J/ L' m; r) X
- //读取ADC的值
0 E C+ w+ x; p3 r1 O - return ADC_GetConversionValue(ADC3);
, ^5 l* i5 ]9 n) t) m" x - 0 {: t3 T2 {* \
- }) Y. [' B. B) C( b2 k
- R) Q3 o& o0 s3 u/ ^* K- //获取某个通道给定次数采样的平均值
" y( r) m) x" h - u16 Get_Adc3_Average(u8 ch, u8 times)6 ^ f% t& [! t0 E) R5 @3 P
- {+ A/ `: `$ _/ l- S) D4 ^
- u32 tmp = 0;( K( w0 j0 r( R: N3 w/ N7 \9 h8 V9 h! O! A
- u8 i = 0;
# f1 q& a: w7 P, ` -
/ m) b& Q' O' p" i% @ - for (i = 0; i < times; i++)* w r; d& {7 `% C7 n6 q8 u
- {
$ A3 w, G: Q% C7 l5 G9 u3 ^ - tmp += Get_Adc3(ch);
_( m, ?3 v8 L) s) M2 f - delay_ms(5);
! C1 D& @# u3 N, u5 n8 b+ W -
9 `+ R0 Z6 w* r* b; @3 m$ k - }
7 v/ \9 \" I- U4 c, Z4 \ - 0 |# ~4 r [/ G: p; \
- //得到平均值9 t6 \% J5 T, O \
- tmp /= times;, o1 E( S' {0 q$ l) X1 h
- 1 h7 ~2 E ]/ K3 c/ }
- if (tmp > 4000)
7 F) O( s' S6 D( X" m - {
" F+ s0 q2 }4 W5 n* f - tmp = 4000;
0 w( z3 h7 n; N! x - }
& U. W9 }. g9 M) \8 `, o - 2 u, w& y; [5 v) i* Q: K
-
5 s$ v/ v3 }1 I - return (u16)(100 - (tmp / 40));& Z8 r- p# o5 A# G L8 E" n
- }
复制代码
: ?( Q" @& h- G+ @- P7 Q& i" ^main.c文件
( E# o: a4 C, Y0 A4 T
! t. k( ]2 q4 l% F# |" @ W- #include "sys.h"
8 R$ z& a& }- Y9 E - #include "delay.h"3 d* S1 o( h5 g
- #include "usart.h"! o7 |. l) r- c
- #include "led.h"
, b6 T5 }% Q/ S" H c; Z' g4 `1 l" I# U - #include "beep.h"( L! G- K5 E; ~: A/ ?8 {. D
- #include "key.h"
1 J% I3 I* R1 [+ m - #include "usmart.h"
" Z8 S. }! {4 ]1 e4 ?- g - #include "lcd.h"
5 j' e- f ~/ P0 ~ - #include "rtc.h"& H1 e. C' u2 j* b$ H6 O% d
- #include "rng.h"* Y4 {$ j$ ^( g) w2 `! w) y5 U
- #include "key.h"$ p2 P: J) c) _8 C
- #include "wkup.h"% f" E% P6 K1 Z' D3 H! Z7 S
- #include "adc.h"
3 Y0 V6 N2 m# c2 g5 G
_6 S: ^/ U$ @- p- int main(void)
8 ?9 h- G3 `) U+ C# C - { / q- ]% ?; Z- H4 X Q" H/ [# n, h
- //u16 value = 0;
$ `" r0 {3 H6 B3 [ -
- S+ C7 S+ ]+ A, | - u16 temp = 0; F' l7 ?2 C+ Z2 U
3 B/ ?3 Y* T1 B3 d- NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2 g B3 z1 S' S7 V
- & S3 T* r+ a: w" a) [
- delay_init(168);
: C. @! ~2 k( F" D; Y - 9 e! q1 p* @( L1 t9 S6 A# i6 X+ M1 |
- uart_init(115200);/ B4 b( s0 f: |: _
-
`( \- o) R& D% ` - //usmart_dev.init(84);
2 P# E4 m A( [5 d t! F* n/ ~ - 6 f, g$ h8 Z# _8 Q7 Q4 L
- 3 B0 S/ n. b) w, M
- LED_Init();
8 K0 W1 O5 u: P3 q6 E -
3 D, w( [2 B% d ^3 O) f; V3 ~ - 6 h& ~( X; P9 w% E5 Z% O. `
- LCD_Init();
w* z3 u% a8 A
?4 H' j- v( T# H- e% M. c9 ~- ADC3_Init();' X* ~' W: C; e) g8 n9 \
-
' L% a; |) g, q) a* g, H. w -
5 z) V: p* u9 U W' h - POINT_COLOR = RED;# X1 U% V& j4 I4 \4 \
- 8 n! |7 l0 A; H4 P- S8 S' }
- LCD_ShowString(30,50,200,16,16,"Explorer STM32F4");
1 V F; e* _6 |9 t, _5 F) I - LCD_ShowString(30,70,200,16,16,"LSENS TEST"); + L' G, C( y) Q* k' F m4 k# M$ z7 K* A
- LCD_ShowString(30,90,200,16,16,"ATOM@tom");$ H& v8 `' ]! k) p9 J. T
- LCD_ShowString(30,110,200,16,16,"2020/09/15");
! C! w- ?+ r5 y3 t% T( n - : A$ q F6 s# h5 h
- //设置字体为蓝色
. X( x! ~7 y# h; \- `- p4 B - POINT_COLOR = BLUE; : A7 D* W" ?* K' y2 x3 K
- LCD_ShowString(30,140,200,16,16,"LSENS VAL:"); . u k8 _7 m' R9 R4 }. H1 P
-
0 k% k. N. j9 U) R - while(1)! X% c w" \- z M) X e {
- {
; V, j, p4 g* e: \ - //得到温度值
. ]* n7 }) X% F- t+ t - temp = Get_Adc3_Average(ADC_Channel_5, 10);
9 @7 U! A, g) c- w6 Y& R9 @ - 9 G' J* w% ~ t6 g3 f
- 1 \2 c3 `. {! a" a; B J
- LCD_ShowxNum(30+10*8,130,temp,3,16,0);//显示ADC的值
( L# D: I8 P) s& L. e4 X4 v) Z
& k! a" k. T$ r, T" w5 `$ I& g$ L- printf("Light: %d\r\n", temp);9 j7 \+ Z$ k9 a6 }) S9 U% h; k5 V
- & j9 x/ d; x: p: L
- LED1 = !LED1;! J. k j* K$ ^; G. C
- ) q% |1 W6 x3 l" R2 m- d
- delay_ms(250);* c: m e/ X: Y& @ w' B+ r
- }/ n; } U5 g( w( ]$ B% z
- }
& M, Y9 {5 G8 C- M5 t
复制代码 ; P2 s3 h! q7 K
04. 结果验证
& U$ x1 L3 ?4 V. v* d$ U& \6 V2 }, Y伴随 DS0 的不停闪烁,提示程序在运行。此时,我们可以通过给 LS1 不同的光照强度,来观察 LSENS_VAL 值的变化,光照越强,该值越大,光照越弱,该值越小。( J1 k& l- T4 t8 b' `& n
3 G6 x: k c2 ?7 z [. C6 v9 n5 M" e
|