2.2 STM32电机测速(正交或者霍尔编码器)9 T7 ~; d# r" [' |/ y% z7 F4 z! H
2.2.1 实现工具1 L8 z. U. F% M3 K6 F
STM32单片机、带编码器的直流减速电机、Keil5、(蓝牙、串口助手)调试用
9 O2 C, A- ^ q$ U) M, `0 a 2.2.2 编码器原理
' L8 M) |& E# s* d7 x9 t1 U(1)编码器是什么?
' ?; x8 i- c2 w) D1 W
编码器是一种将角位移或者角速度转换成一串电数字脉冲的旋转式传感器。编码器又分为光电编码器和霍尔编码器。 " W/ k# F) `/ k$ ~, z
(2)编码器工作原理是什么?
1 J9 A4 e7 M. j5 m( U" z
霍尔编码器是有霍尔马盘和霍尔元件组成。霍尔马盘是在一定直径的圆板上等分的布置有不同的磁极。霍尔马盘与电动机同轴,电动机旋转时,霍尔元件检测输出若干脉冲信号,为判断转向,一般输出两组存在一定相位差的方波信号。示意图如下:
7 c7 E6 M% L! L* Y6 ]4 C3 M
(3)带霍尔编码器的直流减速电机接线图
' {3 ]) V0 o- f, \& I! U8 Q8 U1 \
! }+ E! L9 ? Z2 ` M& R9 [9 S9 y
(4)测速原理:
0 j2 c/ S& J! m. Y/ R$ S+ U
单位时间内,根据脉冲走过的距离计算电机实际速度,这里采用5ms定时器中断。 $ E; g+ d3 Y; f& B
(5)采集数据方式:
) @) P0 p+ C* J* |# [
通常有两种方式,第一种软件技术直接采用外部中断进行采集,根据AB相位差的不同可以判断正负。第二种硬件技术直接使用定时器的编码器模式,这里采用第二种。也是大家常说的四倍频,提高测量精度的方法。
% h, U; H4 K! u. O; k- q7 K
其实就是把AB相的上升沿和下降沿都采集而已,所以1变4。自己使用外部中断方式实现就比较占用资源了,所以不建议使用。 ; g E+ {% W& |
(6)速度计算方法: 1 i" X2 i7 ?, ?
这里计算的是真实的电机轮子的物理转速 电机转动一圈的脉冲数:num1 单位:个 单位时间:t 单位:秒 单位时间内捕获的脉冲变化数:num2 单位:个 (反应电机正反转) 电机轮子半径:r 单位:m 圆周率:pi 单位:无 速度:speed 单位: mm/s
u0 `% b$ ~/ x4 |9 Gspeed = 1000∗num 2∗(2∗pi∗r/num1)/ t speed = 1000*num2*(2*pi*r/num1)/tspeed=1000∗num2∗(2∗pi∗r/num1)/t
: M$ ~/ c, b# x, n 2.2.3 部分代码分享0 N4 Y, I2 v4 w1 _+ I3 ^/ \- Q
! x2 u* `# W- V" i3 m# `
' \& L4 U% K& S) i/ y) `
- _2 ?+ ?8 j' B
主要流程: 将编码器AB相使用的引脚设置成定时器的编码器模式,我们根据TIMx->CNT寄存器数据的变化,计算出单位时间内,脉冲的变化值。然后在定时器中断服务函数中进行速度计算,然后将速度数据通过蓝牙发送到PC的串口助手,验证数据是否正确。 4 [) ~3 }2 R6 q- @+ y
(1)编码器引脚配置以及测速代码
( J4 q# c7 W0 X: G# {' M, ~% ^0 l# `+ g- #include "encoder.h"
: |* a4 F$ Z( b8 S$ p6 F
( Q. J( ]! }5 {0 x- /**************************************************************************" l* _2 h6 |3 E8 @4 m( Z
- 函数功能:把TIM2初始化为编码器接口模式6 I5 ~ X. F$ P% |0 o
- 入口参数:无
- ~) f( P; X0 L z/ @/ q0 [+ ~ - 返回 值:无
% h+ {+ R" X& r' L8 U x - **************************************************************************/( U3 J. c; ~0 k/ j, d R* M' v# p
- void Encoder_Init_TIM2(void)
" |9 b: h4 {( ]# z! w4 V# I8 s - {$ o o9 |! a4 ]% t5 M+ K
- TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
/ N* M/ o/ r6 z0 ^ - TIM_ICInitTypeDef TIM_ICInitStructure;
. G; G; X& O$ j$ ^# b* P: n$ c - GPIO_InitTypeDef GPIO_InitStructure; k" }/ S9 R9 A2 k9 v
- RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); //使能定时器2的时钟4 I' p+ R+ C0 ~, R0 i* k. K
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能PA端口时钟" _/ u1 n5 l3 [& c/ A
- t9 ?; o8 t0 N- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1; //端口配置
8 s; B# {: i% Q+ H" p - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入3 a" m; i D8 l2 ?# ]
- GPIO_Init(GPIOA, &GPIO_InitStructure); //根据设定参数初始化GPIOA$ t5 h; N. G b; d0 Q
- 2 T3 G8 d; {$ ]' Q5 F9 q: e; ^
- TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
; _1 F; D1 `' D/ m! S% ] - TIM_TimeBaseStructure.TIM_Prescaler = 0x0; //预分频器
5 V) G0 y! J* t6 i* {, j' i5 ? - TIM_TimeBaseStructure.TIM_Period = ENCODER_TIM_PERIOD; //设定计数器自动重装值5 j7 H4 k; p- u( w7 W! d
- TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //选择时钟分频:不分频6 Y7 r% [% ?9 j5 `6 R4 W
- TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM向上计数 2 h4 r+ k' B) d( T, C3 f
- TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
1 n% @+ p4 |8 Y, C/ u" p5 a% d$ O -
2 w7 H; k% S! D1 _/ l8 b+ N i - TIM_EncoderInterfaceConfig(TIM2, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);//使用编码器模式33 D ?. R- q' V5 J! ~1 E7 b
- ' G% ~% z1 v z. K1 a) l
- TIM_ICStructInit(&TIM_ICInitStructure);
; H" X& E3 X) n& D7 r/ } - TIM_ICInitStructure.TIM_ICFilter = 10;, g+ W8 _9 U, d3 y' ~ K, K z
- TIM_ICInit(TIM2, &TIM_ICInitStructure);
8 c# s ]9 Z \6 n5 R, y - TIM_ClearFlag(TIM2, TIM_FLAG_Update); //清除TIM的更新标志位
* A, Q% o! O; f' {0 q8 C - TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);
h' t& J5 U* @- f) l - //Reset counter
9 c+ S+ C N; m7 t& {# s - TIM_SetCounter(TIM2,0);+ a' o& ~' t3 h1 \- p
- //===============================================6 ]! t' l$ Y; d) ~
- TIM2->CNT = 0x7fff;) [* }: _6 ~( v! r1 a+ V
- //===============================================
- g' s0 {( I& t& h' v, p - TIM_Cmd(TIM2, ENABLE); ) I1 O6 r" F- {2 t( \
- }
8 n F2 s' i5 r! w1 { - /**************************************************************************
) g- B! I7 f7 f1 J/ J7 p" T - 函数功能:把TIM4初始化为编码器接口模式
$ @, M' n( {& U! {: a0 B ~ - 入口参数:无4 w2 z$ y4 Z6 p+ o) }1 S
- 返回 值:无1 c4 a" K4 A: G- ]
- **************************************************************************/
' V- A; ]; i* h. p- `& s - void Encoder_Init_TIM4(void)
6 b1 D7 [' x: J Y4 p - {- a( l1 J! l4 X! _% q9 ?5 S
- TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
) b% E7 w0 Q. B/ m( i8 r - TIM_ICInitTypeDef TIM_ICInitStructure; / l' \4 U( P; y: S3 n% g
- GPIO_InitTypeDef GPIO_InitStructure;
" I# O8 b( Q7 j2 ]; N - RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE); //使能定时器4的时钟+ O7 x# K8 v \6 _6 ]0 ~4 \! i3 h
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能PB端口时钟9 G* i$ o6 L A
. M* u5 K9 z4 K- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7; //端口配置
+ m8 J6 V) c5 p, A/ E- w' T' F - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入5 U4 \% V/ w: l0 G# [
- GPIO_Init(GPIOB, &GPIO_InitStructure); //根据设定参数初始化GPIOB x! I( S: a! c' s- ^' ]3 R" i
- - p8 z' q8 e9 G. Q5 |! N
- TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);5 O* y( W6 G/ p6 t
- TIM_TimeBaseStructure.TIM_Prescaler = 0x0; // 预分频器
$ ^8 b' `/ T2 Y8 F - TIM_TimeBaseStructure.TIM_Period = ENCODER_TIM_PERIOD; //设定计数器自动重装值9 x, o8 n- X. w8 N6 ?6 A% V* n! p
- TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //选择时钟分频:不分频
2 v+ T) a4 Q$ w - TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM向上计数
4 D2 G: O' W+ c5 W& B - TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);
: T8 m$ `; C) m5 E$ y - TIM_EncoderInterfaceConfig(TIM4, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);//使用编码器模式33 L3 c3 A C. M
- TIM_ICStructInit(&TIM_ICInitStructure);
7 B$ V$ {* A/ e) p0 d* u - TIM_ICInitStructure.TIM_ICFilter = 10;
3 E6 c$ x T9 f: h" Z$ m - TIM_ICInit(TIM4, &TIM_ICInitStructure);" A- @' X9 F! }, N
- TIM_ClearFlag(TIM4, TIM_FLAG_Update); //清除TIM的更新标志位
: |+ f5 D8 h. z' P1 w - TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE);
B: K/ _1 }( n - //Reset counter
; J* K+ i2 X1 L2 l+ d - TIM_SetCounter(TIM4,0);4 X' ?8 f: j: B& L* [
- //===============================================
, q/ o, l1 \' y: z) {, B - TIM4->CNT = 0x7fff;
3 ^1 g9 _3 b7 U3 a - //===============================================% H, Z. T- p0 l3 `# q- l
- TIM_Cmd(TIM4, ENABLE); 9 ?) m; m) L8 |
- }
3 M6 {9 U6 Y1 k$ @$ C
6 q: O* o, W0 @4 E E- /**************************************************************************
5 b! J, P2 h0 L6 u. Q$ K9 c - 函数功能:读取编码器脉冲差值6 y6 A; {$ H5 x5 K( M) }3 T3 ]
- 入口参数:TIM_TypeDef * TIMx2 g- D4 {% C4 ?: e* |4 i
- 返回 值:无- ^0 a: Z2 f& R+ `' B7 J" b
- **************************************************************************/
& E6 g( _3 r: k( O# @" u8 v - s16 getTIMx_DetaCnt(TIM_TypeDef * TIMx)$ r5 }9 _5 Q% @. n7 Q
- {9 @, C8 E( m" n
- s16 cnt;. C* B* P2 L8 {- x7 h- H( I. i, f! P( D
- cnt = TIMx->CNT-0x7fff; //这一点默认认为,单位时间内编码器不会出现0x7fff的脉冲变化,事实上也是这样
- P! q; w) {' P, s- I- i% P B - TIMx->CNT = 0x7fff;5 n4 f3 i. b# N, x1 R
- return cnt;
! Y6 w0 B7 y/ a0 s - }
# ~1 f' t0 ~- h: m; n
- s$ r7 X: L( k7 {, x+ N. l- /**************************************************************************6 H9 D d" O, W6 ]7 Z
- 函数功能:计算左右轮速, z) `1 p1 w, |, X( j- g2 q
- 入口参数:int *leftSpeed,int *rightSpeed g5 Q+ A# a: L
- 返回 值:无9 C; I8 Z7 d, _7 q% p
- //计算左右车轮线速度,正向速度为正值 ,反向速度为负值,速度为乘以1000之后的速度 mm/s
5 B1 v6 W0 E+ h - //一定时间内的编码器变化值*转化率(转化为直线上的距离m)*200s(5ms计算一次) 得到 m/s *1000转化为int数据9 u& `7 d( S" t
2 d. l& \. D; e" ?- 一圈的脉冲数:8 R( f2 F7 W) A8 F6 M3 I9 @
- 左:1560
- u# C+ R) u6 q6 p2 }1 g - 右:1040
7 s/ p! A; f y" [2 i, d - 轮子半径:0.03m
( O( I# Z7 e( Z0 L - 轮子周长:2*pi*r
" d* @- z3 A; [8 |! X( d1 o0 w# } - 一个脉冲的距离:
2 X. y3 W: X* @% R+ p+ @ - 左:0.000120830m; g) h( _1 o& D3 [3 h4 _9 e" U; W
- 右:0.000181245m9 O8 Q j5 O5 w7 p# B( S: k
- 速度分辨率: x) Q5 o: `3 Z# c" {' P. _
- 左:0.0120m/s 12.0mm/s
; n' P' k4 H. h3 a) a - 右:0.0181m/s 18.1mm/s# `# o( S% f# T9 D2 _- r
- **************************************************************************/
+ i S& D) a7 W( w: h - - U/ N7 y7 l. m; p
- void Get_Motor_Speed(int *leftSpeed,int *rightSpeed)
2 [3 ]" k& E) E; Y: x - {
3 |) G. S, K5 ]% ~& e" T, P5 I - static int leftWheelEncoderNow = 0;# I& o3 c) g3 h" z
- static int rightWheelEncoderNow = 0;+ R/ W; [4 s8 e; d `
- static int leftWheelEncoderLast = 0;
, p. {2 J' B( [0 T - static int rightWheelEncoderLast = 0; 5 \' ~' v3 r8 x( E9 f
-
$ U/ h8 o0 E" h - //记录本次左右编码器数据
- q% C) `& y7 Q, S9 Z! ^ - leftWheelEncoderNow += getTIMx_DetaCnt(TIM4);
' b- _: z$ n/ E6 a' K' x - rightWheelEncoderNow+= getTIMx_DetaCnt(TIM2);2 S! _1 I8 K: M- Q( Z
-
7 q- Q: |& \% Y! C4 o - //5ms测速
0 ?( x/ {, L7 R( m! ~5 s) G! j5 s - *leftSpeed = (leftWheelEncoderNow - leftWheelEncoderLast)*1000*200*0.000120830; + G( G E% p5 G% K$ q$ z. Q
- *rightSpeed = (rightWheelEncoderNow - rightWheelEncoderLast)*1000*200*0.000181245;
, y3 Z0 F. _1 k M# [* q
, |, D4 G% [% ^- //记录上次编码器数据4 g. x# V( R) t8 [. [: X9 s
- leftWheelEncoderLast = leftWheelEncoderNow;
W( C+ R) b- m - rightWheelEncoderLast = rightWheelEncoderNow; 2 _# m5 H7 Q$ [5 h) ?8 }$ \
- }
复制代码(2)main.c ' X5 {) [1 _6 L0 @) J( e! g
- #include "sys.h"
! C& ?8 Z3 M9 V7 V c8 p1 I - 8 i# n2 q: M8 w- k) S
- //====================自己加入的头文件===============================& r: \7 ~3 w/ i$ h1 Z, N: F
- #include "delay.h"
. g# | u0 B2 _/ w& v1 ~2 |- Y. k - #include "led.h"
3 m0 U+ x, {4 Z) L - #include "encoder.h"
/ B, M: ?$ Q( g5 Z - #include "usart3.h"
; S) K, q. S/ _7 |6 ?0 w& u V - #include "timer.h"
& ]/ u; ^0 I- J1 w. L( z0 j* i: ? - #include <stdio.h>
1 C. r* G8 w3 X0 F: `6 o - //===================================================================
' t4 s/ S, S5 D& n" Y; H9 I - 5 T& c2 X3 H2 `. m2 `
- int leftSpeedNow =0;
1 t; M/ F) J. Z4 E2 q9 F1 o - int rightSpeedNow =0;
s5 s$ l4 ~! p5 W- H* Z - G9 v4 `9 l; O* T( T/ ?& n; J, Q
- int main(void)4 R. m1 p% w7 J, M. z
- {
' O! {6 d& c9 [8 a
/ e6 Q6 ~ [4 A- GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable,ENABLE);
! m- m. u/ x# D: Q- N- [6 L - GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);//禁用JTAG 启用 SWD
; }& w5 m5 Y% X _; _$ M9 W* D -
: p9 `. f6 s* X+ Q: @7 A - MY_NVIC_PriorityGroupConfig(2); //=====设置中断分组
( v# ]# j% b+ I$ ~# ?- c. h2 p - 6 X. B+ M2 F& t X& P
- delay_init(); //=====延时函数初始化
) w$ E; W- c2 @- X; F - LED_Init(); //=====LED初始化 程序灯 " j6 _7 `. n. r4 r* q, r
- 4 _+ U: \: c4 M; m* N0 d9 k
- usart3_init(9600); //=====串口3初始化 蓝牙 发送调试信息
2 Z9 i# l3 l. t, _ - + T+ o% z5 H* X$ X9 D4 x5 p. u4 m3 s4 f
- Encoder_Init_TIM2(); //=====初始化编码器1接口* V! T( ?( \7 L# x. s- p- \$ b" t
- Encoder_Init_TIM4(); //=====初始化编码器2接口 E+ G& ^+ I0 A. v* V
- 8 u Q0 k1 S' ]
- TIM3_Int_Init(50-1,7200-1); //=====定时器初始化 5ms一次中断9 I" L/ f1 g! \( C( y' ?* \
-
) ]1 E2 o2 u1 _ - while(1)+ {3 t( I- f3 z% H- F1 D' J$ B
- { a9 e. e! T) E1 [! U# z# n: I
- printf("L=%d,R=%d\r\n",leftSpeedNow,rightSpeedNow);
; u/ S6 G$ y; p0 z: T. C& ~; v- { - delay_ms(15);0 g9 z1 N H$ `3 F: _* H9 G# S, Q
- } ! q; H0 i7 A" k f& M
- }
/ m( a% T2 j4 K( B - . X, D' N- S1 A+ r6 C; q
- //5ms 定时器中断服务函数
+ M+ |2 P2 D- k2 j% t - : T7 u! G$ {6 [9 r' S# o
- void TIM3_IRQHandler(void) //TIM3中断
. @7 A3 F1 S+ E: l/ ? - { j0 j/ z: H9 O4 N* F
- if(TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否:TIM 中断源
( a0 m6 U. d; g- F# T - {) L) @0 Q& C3 V; M
- TIM_ClearITPendingBit(TIM3, TIM_IT_Update); //清除TIMx的中断待处理位:TIM 中断源
" K$ S$ T1 c: @. C9 m - ! ^1 Z9 p1 p: | `9 R
- Get_Motor_Speed(&leftSpeedNow,&rightSpeedNow);5 h9 v- a" V# Y: H8 B
- Led_Flash(100);
8 X5 y: R, o3 P& `, Z - } o1 ~# A" s# N5 w. s. s+ G
- }
复制代码 9 G K9 r; Y; X8 n9 k
; q/ W) w$ t1 ~ _3 R
, {+ Q9 @ M/ k) S
2.2.4 总结上面的一篇文章,我们实现的STM32单片机对直流减速电机的测速。再加上之前介绍的电机PWM控制,下一篇我们要结合前两篇的内容加上PID实现对电机速度的闭环控制。 2 Z: P3 I8 d& N" O" D8 C/ b
8 J) @" j: a- a @! d6 ]- g
2 n, @, F5 H l2 H
|
博主写的教程很清楚,我写了20多篇关于编码器测速+STM32CubeMX开发+GPIO、ADC采集+PID电机控制+openmv手把手开发的教程欢迎大家一起交流:这是文章:https://blog.csdn.net/qq_46187594/article/details/138110155?spm=1001.2014.3001.5502 这个是视频:https://www.bilibili.com/video/BV16x4y1M7EN/?spm_id_from=333.337.search-card.all.click