旋转编码器的实物图和结构图及程序
数字旋转编码器在现在的工控领域运用广泛,跑步机的旋钮,控制器的旋钮,音响的音量控制旋钮,都是数字旋转编码器的典型应用。
5脚的数字旋转编码器:
图 1 旋转编码器的实物图
具有左转,右转,按下三个功能。4、5脚是中间按下去的开关接线 1 2 3脚 一般是中间2脚接地,1、3脚上拉电阻后,当左转、右转旋纽时,在1、3脚就有脉冲信号输出了。
着这是标准资料:
图2 旋转编码器的结构图
在单片机编程时,左转和右转的判别是难点,用示波器观察这种开关左转和右转时两个输出脚的信号有个相位差,见下图:
图3 旋转编码器在旋转的时候的输出波形
由此可见,如果输出1为高电平时,输出2出现一个高电平,这时开关就是向顺时针旋转; 当输出1 为高电平,输出2出现一个低电平,这时就一定是逆时针方向旋转。
所以,在单片机编程时只需要判断当输出1为高电平时,输出2当时的状态就可以判断出是左旋转或是右旋转了。
还有另外一种3脚的,除了不带按钮开关外,和上面是一样的使用。
图4 三个脚的旋转编码器
参考程序:
#include reg51.h
#define uint unsigned int
#define uchar unsigned char
sbit Rotation_Key_A = P1_1; //定义旋转编码器的方向判断A口
sbit Rotation_Key_B = P1_2; //定义旋转编码器的方向判断B口
uint CodingsWitchPolling()//
{
static uchar A_Last_State,B_Last_State; //定义了两个变量用来储蓄上一次调用此方法是编码开关两引脚的电平
static uchar High_Last_State;//定义了一个变量用来储蓄以前是否出现了两个引脚都为高电平的状态
uint tmp = 0;
if(Rotation_Key_A&&Rotation_Key_B)
High_Last_State = 1; //
if(High_Last_State) //如果High_Last_State为1执行下面的步骤
{
if(Rotation_Key_A==0&&Rotation_Key_B==0) //如果当前编码开关的两个引脚都为底电平执行下面的步骤
{
if(B_Last_State) //为高说明编码开关在向加大的方向转
{
High_Last_State = 0;
tmp++; //
}
if(A_Last_State) //为高说明编码开关在向减小的方向转
{
High_Last_State = 0;
tmp--; //设返回值
}
}
}
A_Last_State = Rotation_Key_A; //储存A口的当前状态
B_Last_State = Rotation_Key_B; //储存B口的当前状态
return tmp; //
}
//编码器计数程序
void encoder_cnt(void)
{
uchar Reade_Io_Data;
Reade_Io_Data = PIND; //取端口D管脚信号
Reade_Clr_Signal = (Reade_Io_Data & 0x08); //读取编码器清零信号
if(Reade_Clr_Signal != false) //有编码器清零信号
{
couch_num = 0; //水平床码清零
}
else
{
if(encoder_cnt_en == false) //编码器计数模块没有启动
{
pr_couch_ba = Reade_Io_Data & 0x03; //取编码器A、B相电平信号
}
else
{
couch_ba = Reade_Io_Data & 0x03; //取编码器A、B相电平信号
if(pr_couch_ba == 0x00)
{
if(couch_ba == 0x01)
{
couch_num++; //水平床码加1
}
else if(couch_ba == 0x10)
{
couch_num--; //水平床码减1
}
}
else if(pr_couch_ba == 0x01)
{
if(couch_ba == 0x11)
{
couch_num++; //水平床码加1
}
else if(couch_ba == 0x00)
{
couch_num--; //水平床码减1
}
}
else if(pr_couch_ba == 0x10)
{
if(couch_ba == 0x00)
{
couch_num++; //水平床码加1
}
else if(couch_ba == 0x11)
{
couch_num--; //水平床码减1
}
}
else if(pr_couch_ba == 0x11)
{
if(couch_ba == 0x10)
{
couch_num++; //水平床码加1
}
else if(couch_ba == 0x01)
{
couch_num--; //水平床码减1
}
}
}
pr_couch_ba = couch_ba;
}
}
编码器及其计数模块原理
旋转编码器的检测程序(基于51单片机)
//旋转编码器检测程序,A/B信号分别接在了INT0和INT1上
//2005年3月27用KEIL编译、硬件测试通过
//注意:编码器的信号,程序未做消抖处理。测试中,A/B信号上各
//接了一只104的瓷片电容,工作很正常。如果不接电容,请自行编
//写信号消抖程序。
#include
sbit led=0xB1;//有一只LED接在了RXD引脚上,用来指示正反转;
main()
{
EA=1; //总中断允许
EX0=1; //外部中断0允许
IT0=1; //外部中断0为边沿触发方式
while(1);;
}
/*********************
编码器中断函数
入口:无
出口:无
*********************/
void encoder(void) interrupt 0
{ //外部中断0
if (INT1)
{
led=1;
}
else
{
led=0;
}
}
whimsy 的AVR程序
//外部中断0,用于编码开关解码,解码图: A接中断脚(AVR的PD2),以此为基准,B用来判断方向(连到AVR的PA1), C接地
//A -|
// | -----|__________|----------|____________
//C -|
//
//B -|
// | ----------|__________|----------|____________
//C -|
// CW ===>>> ROTATION
//外部中断设置(ISC01=0,ISC00=1): INT0 引脚上任意的逻辑电平变化都将引发中断
#pragma interrupt_handler int0_isr:2
void int0_isr(void)
{
//external interupt on INT0
GICR=0; //禁止外部中断
if ((PIND & 0x04)==0) //先判断是高电平产生的中断还是低点平的中断
{
if ((PINA & 0x02)==0) //再判断B线上的电平
{
keycounter--;
keydirection=0;
}
else
{
keycounter++;
keydirection=1;
}
else
if ((PINA & 0x02)==0)
{
keycounter++;
keydirection=1;
}
else
{
keycounter--;
keydirection=0;
}
GICR=0x40;
}
}