实用的电机控制程序

发布者:快乐的成长最新更新时间:2016-12-24 来源: eefocus关键字:电机控制程序  单片机 手机看文章 扫描二维码
随时随地手机看文章

上面我们虽然完成了用中断控制电机转动的程序,但实际上这个程序还是没多少实用价值的,我们不能每次想让它转动的时候都上下电啊,是吧。还有就是它不但能正转还得能反转啊,也就是说不但能转过去,还得能转回来呀。好吧,我们就来做一个实例程序吧,结合第 8 章的按键程序,我们设计这样一个功能程序:按数字键 1~9,控制电机转过 1~9 圈;配合上下键改变转动方向,按向上键后正向转 1~9 圈,向下键则反向转 1~9 圈;左键固定正转 90 度,右键固定反转 90;Esc 键终止转动。通过这个程序,我们也可以进一步体会到如何用按键来控制程序完成复杂的功能,以及控制和执行模块之间如何协调工作,而你的编程水平也可以在这样的实践练习中得到锻炼和提升。

#include

sbit KEY_IN_1 = P2^4;

sbit KEY_IN_2 = P2^5;

sbit KEY_IN_3 = P2^6;

sbit KEY_IN_4 = P2^7;

sbit KEY_OUT_1 = P2^3;

sbit KEY_OUT_2 = P2^2;

sbit KEY_OUT_3 = P2^1;

sbit KEY_OUT_4 = P2^0;

unsigned char code KeyCodeMap[4][4] = { //矩阵按键编号到标准键盘键码的映射表

    { 0x31, 0x32, 0x33, 0x26 }, //数字键 1、数字键 2、数字键 3、向上键

    { 0x34, 0x35, 0x36, 0x25 }, //数字键 4、数字键 5、数字键 6、向左键

    { 0x37, 0x38, 0x39, 0x28 }, //数字键 7、数字键 8、数字键 9、向下键

    { 0x30, 0x1B, 0x0D, 0x27 } //数字键 0、ESC 键、 回车键、 向右键

};

unsigned char KeySta[4][4] = { //全部矩阵按键的当前状态

    {1, 1, 1, 1}, {1, 1, 1, 1}, {1, 1, 1, 1}, {1, 1, 1, 1}

};

signed long beats = 0; //电机转动节拍总数

void KeyDriver();

void main(){

    EA = 1; //使能总中断

    TMOD = 0x01; //设置 T0 为模式 1

    TH0 = 0xFC; //为 T0 赋初值 0xFC67,定时 1ms

    TL0 = 0x67;

    ET0 = 1; //使能 T0 中断

    TR0 = 1; //启动 T0

   

    while (1){

        KeyDriver(); //调用按键驱动函数

    }

}

/* 步进电机启动函数,angle-需转过的角度 */

void StartMotor(signed long angle){

    //在计算前关闭中断,完成后再打开,以避免中断打断计算过程而造成错误

    EA = 0;

    beats = (angle * 4076) / 360; //实测为 4076 拍转动一圈

    EA = 1;

}

/* 步进电机停止函数 */

void StopMotor(){

    EA = 0;

    beats = 0;

    EA = 1;

}

/* 按键动作函数,根据键码执行相应的操作,keycode-按键键码 */

void KeyAction(unsigned char keycode){

    static bit dirMotor = 0; //电机转动方向

    //控制电机转动 1-9 圈

    if ((keycode>=0x30) && (keycode<=0x39)){

        if (dirMotor == 0){

            StartMotor(360*(keycode-0x30));

        }else{

            StartMotor(-360*(keycode-0x30));

        }

    }else if (keycode == 0x26){ //向上键,控制转动方向为正转

        dirMotor = 0;

    }else if (keycode == 0x28){ //向下键,控制转动方向为反转

        dirMotor = 1;

    }else if (keycode == 0x25){ //向左键,固定正转 90 度

        StartMotor(90);

    }else if (keycode == 0x27){ //向右键,固定反转 90 度

        StartMotor(-90);

    }else if (keycode == 0x1B){ //Esc 键,停止转动

        StopMotor();

    }

}

/* 按键驱动函数,检测按键动作,调度相应动作函数,需在主循环中调用 */

void KeyDriver(){

    unsigned char i, j;

    static unsigned char backup[4][4] = { //按键值备份,保存前一次的值

        {1, 1, 1, 1}, {1, 1, 1, 1}, {1, 1, 1, 1}, {1, 1, 1, 1}

    };

   

    for (i=0; i<4; i++){ //循环检测 4*4 的矩阵按键

        for (j=0; j<4; j++){

            if (backup[i][j] != KeySta[i][j]){ //检测按键动作

                if (backup[i][j] != 0){ //按键按下时执行动作

                    KeyAction(KeyCodeMap[i][j]); //调用按键动作函数

                }

                backup[i][j] = KeySta[i][j]; //刷新前一次的备份值

            }

        }

    }

}

/* 按键扫描函数,需在定时中断中调用,推荐调用间隔 1ms */

void KeyScan(){

    unsigned char i;

    static unsigned char keyout = 0; //矩阵按键扫描输出索引

   

    static unsigned char keybuf[4][4] = { //矩阵按键扫描缓冲区

        {0xFF, 0xFF, 0xFF, 0xFF}, {0xFF, 0xFF, 0xFF, 0xFF},

        {0xFF, 0xFF, 0xFF, 0xFF}, {0xFF, 0xFF, 0xFF, 0xFF}

    };

   

    //将一行的 4 个按键值移入缓冲区

    keybuf[keyout][0] = (keybuf[keyout][0] << 1) | KEY_IN_1;

    keybuf[keyout][1] = (keybuf[keyout][1] << 1) | KEY_IN_2;

    keybuf[keyout][2] = (keybuf[keyout][2] << 1) | KEY_IN_3;

    keybuf[keyout][3] = (keybuf[keyout][3] << 1) | KEY_IN_4;

    //消抖后更新按键状态

    for (i=0; i<4; i++){ //每行 4 个按键,所以循环 4 次

        if ((keybuf[keyout][i] & 0x0F) == 0x00){

            //连续 4 次扫描值为 0,即 4*4ms 内都是按下状态时,可认为按键已稳定的按下

            KeySta[keyout][i] = 0;

        }else if ((keybuf[keyout][i] & 0x0F) == 0x0F){

            //连续 4 次扫描值为 1,即 4*4ms 内都是弹起状态时,可认为按键已稳定的弹起

            KeySta[keyout][i] = 1;

        }

    }

    //执行下一次的扫描输出

    keyout++; //输出索引递增

    keyout = keyout & 0x03; //索引值加到 4 即归零

    //根据索引,释放当前输出引脚,拉低下次的输出引脚

    switch (keyout){

        case 0: KEY_OUT_4 = 1; KEY_OUT_1 = 0; break;

        case 1: KEY_OUT_1 = 1; KEY_OUT_2 = 0; break;

        case 2: KEY_OUT_2 = 1; KEY_OUT_3 = 0; break;

        case 3: KEY_OUT_3 = 1; KEY_OUT_4 = 0; break;

        default: break;

    }

}

/* 电机转动控制函数 */

void TurnMotor(){

    unsigned char tmp; //临时变量

    static unsigned char index = 0; //节拍输出索引

    unsigned char code BeatCode[8] = { //步进电机节拍对应的 IO 控制代码

        0xE, 0xC, 0xD, 0x9, 0xB, 0x3, 0x7, 0x6

    };

   

    if (beats != 0){ //节拍数不为 0 则产生一个驱动节拍

        if (beats > 0){ //节拍数大于 0 时正转

            index++; //正转时节拍输出索引递增

            index = index & 0x07; //用&操作实现到 8 归零

            beats--; //正转时节拍计数递减

            }else{ //节拍数小于 0 时反转

            index--; //反转时节拍输出索引递减

            index = index & 0x07; //用&操作同样可以实现到-1 时归 7

            beats++; //反转时节拍计数递增

        }

        tmp = P1; //用 tmp 把 P1 口当前值暂存

        tmp = tmp & 0xF0; //用&操作清零低 4 位

        tmp = tmp | BeatCode[index]; //用|操作把节拍代码写到低 4 位

        P1 = tmp; //把低 4 位的节拍代码和高 4 位的原值送回 P1

    }else{ //节拍数为 0 则关闭电机所有的相

        P1 = P1 | 0x0F;

    }

}

/* T0 中断服务函数,用于按键扫描与电机转动控制 */

void InterruptTimer0() interrupt 1{

    static bit div = 0;

    TH0 = 0xFC; //重新加载初值

    TL0 = 0x67;

    KeyScan(); //执行按键扫描

    //用一个静态 bit 变量实现二分频,即 2ms 定时,用于控制电机

    div = ~div;

    if (div == 1){

        TurnMotor();

    }

}

这个程序是第 8 章和本章知识的一个综合——用按键控制步进电机转动。程序中有这么几点值得注意,我们分述如下:

针对电机要完成正转和反转两个不同的操作,我们并没有使用正转启动函数和反转启动函数这么两个函数来完成,也没有在启动函数定义的时候增加一个形式参数来指明其方向。我们这里的启动函数 void StartMotor(signed long angle)与单向正转时的启动函数唯一的区别就是把形式参数 angle 的类型从 unsigned long 改为了 signed long,我们用有符号数固有的正负特性来区分正转与反转,正数表示正转 angle 度,负数就表示反转 angle 度,这样处理是不是很简洁又很明了呢?而你对有符号数和无符号数的区别用法是不是也更有体会了?

针对终止电机转动的操作,我们定义了一个单独的 StopMotor 函数来完成,尽管这个函数非常简单,尽管它也只在 Esc 按键分支内被调用了,但我们仍然把它单独提出来作为了一个函数。而这种做法就是基于这样一条编程原则:尽可能用单独的函数来完成硬件的某种操作,当一个硬件包含多个操作时,把这些操作函数组织在一起,形成一个对上层的统一接口。这样的层次化处理,会使得整个程序条理清晰,既有利于程序的调试维护,又有利于功能的扩充。

中断函数中要处理按键扫描和电机驱动两件事情,而为了避免中断函数过于复杂,我们就又分出了按键扫描和电机驱动两个函数(这也同样符合上述 2 的编程原则),而中断函数的逻辑就变得简洁而清晰了。这里还有个矛盾,就是按键扫描我们选择的定时时间是 1ms,而本章之前的实例中电机节拍持续时间都是 2ms;很显然,用 1ms 的定时可以定出 2ms 的间隔,而用 2ms 的定时却得不到准确的 1ms 间隔;所以我们的做法就是,定时器依然定时 1ms,然后用一个 bit 变量做标志,每 1ms 改变一次它的值,而我们只选择值为 1 的时候执行一次动作,这样就是 2ms 的间隔了;如果我要 3ms、4ms„„呢,把 bit 改为 char 或 int 型,然后对它们递增,判断到哪个值该归零,就可以了。这就是在硬件定时器的基础上实现准确的软件定时,其实类似的操作我们在讲数码管的时候也用过了,回想一下吧。


关键字:电机控制程序  单片机 引用地址:实用的电机控制程序

上一篇:28BYJ-48步进电机控制程序基础
下一篇:单片机蜂鸣器控制程序和驱动电路

推荐阅读最新更新时间:2024-03-16 15:26

基于ADC081S051与51单片机的接口电路及驱动程序
引言 自然界存在的信号通常为模拟信号,在测控系统中通常将采集到的模拟信号数字化,然后交给微处理器或微控制器进行处理。因此模数转换器成为测控系统中不可缺少的部分。ADC081S051 模数转换器是国家半导体公司生产的低功率单通道CMOS 8位A/D转换器,它采用串行外设接口方式进行数据输出。与传统的器件不同的是,传统器件的采样速率是固定的,而ADC081S051 的采样速率可以在200~500 KSPS 范围内可变。 ADC081S051 的数据输出是串行数据输出,它与SPI、QSPI 等一些标准相兼容。它的工作电压范围在2.7~5.25 V,因此可以采用3 V或者5 V电源供电。它可以应用于便携式系统、远程数据采集、仪器与控制系统
[电源管理]
基于ADC081S051与51<font color='red'>单片机</font>的接口电路及驱动程序
51单片机(使用8x8点阵)
1.说在前面 今天初步学会了怎样使用面包板,感觉挺好玩的,学硬件的路渐行渐远吧 2.74hc595模块(8位移位寄存器和一个存储器,三态输出功能) 1.将三位串行输入变为并行输出 2.数据先移位到最高点在移位到次高点 3.基础参考资料:https://wenku.baidu.com/view/4d619b1c650e52ea55189826.html 3.实现点亮8x8点阵的代码 #include reg52.h typedef unsigned char u8; typedef unsigned int u16; //定义串行端口 sbit SRCLK=P3^6; sbit RCLK=P3^
[单片机]
51<font color='red'>单片机</font>(使用8x8点阵)
51单片机手机拨号模块
#include AT89X52.h #include intrins.h #define Delaynop(){_nop_();_nop_();_nop_();_nop_();} sbit RS=P2^0; sbit RW=P2^1; sbit EN=P2^2; void Delayms(unsigned int ms); bit LCD_Busy(); void LCD_Pos(unsigned char); void LCD_Wcmd(unsigned char); void LCD_Wdat(unsigned char); char code Title_Text ={ --Phone Code--
[单片机]
51<font color='red'>单片机</font>手机拨号模块
ATMEL单片机、CPLD下载线硬件简介
介绍 本应用文档描述了 Atmel? ATMEL 单片机 、CPLD下载线接口;使用 AT89SISP 编程器给 AT89SX微控制器在线编程的接口。 AT89ISP 软件Atmel 的在线编程软件已经在我们的网站上提供了下载,具体的操作方法请在软件说明中查询(提供下载),软件可以在 Windows? 9x/ME/2000/NT? 操作系统上运行。Atmel ISP 软件提供可以编写的微控制器的芯片有以下几种:AT89S51、AT89S52、AT89S53、AT89LS53、AT89LS8252、AT89LS8252等几种。可以提供用户目标板使用的晶振输入(用户使用的晶振可能不太一样)。 AT89ISP 电缆 为了使用At
[单片机]
ATMEL<font color='red'>单片机</font>、CPLD下载线硬件简介
基于PIC16F84A单片机编辑LED数码管秒显示器的程序
  源程序pIC07.c是基于PIC16F84A单片机编辑的2位LED数码管秒显示器,其显示方式为0~99秒重复显示。如果读者需要进行实验,可在硬件电路上进行实时显示(当然,前提是先将该程序对应的目标码.hex文件烧写到PIC16F84A芯片中)。   编辑2位LED数码管秒显示程序pic07.c的目的,是为了方便介绍对程序进行模拟调试的方法和操作过程。   完整的源程序pic07.c清单如下:
[单片机]
基于PIC16F84A<font color='red'>单片机</font>编辑LED数码管秒显示器的程序
富士通半导体推出宽电压双Flash MCU MB95650系列
富士通半导体(上海)有限公司日前宣布推出采用新工艺的宽工作电压、12bit高精度ADC、且带主从I2C控制器的高性价比双FLASH通用8位微控制器MB95560系列。该系列产品包括采用SOP24、TSSOP24、QFN32三种封装形式的24款产品。2012年6月下旬开始提供样片,2012年9月开始批量供货。 MB95650系列以8位微控制器F2MC-New8FX家族为基础,采用宽工作电压设计,能在1.8V~5.5V的应用环境中工作。该系列产品集成了主从I2C控制器和12bit高精度AD转换器,并采用双通道FLASH技术,使得程序在FLASH上运行的同时可以对另一通道FLASH进行读写,从而保护客户所需的重要数据。该系列产品还集成
[单片机]
V2XE型地磁传感器在单片机系统中的应用
摘要:在GPS导航定位系统中,地磁方位是一个很重要的参数。V2XE是PNI公司推出的基于SPI接口的地磁传感器。分析了其工作原理和指令集,并结合实际介绍了V2XE在AT89C2051型单片机系统中的硬件连接和软件编制。结果表ql,该系统测量地磁方向和温度的精度较高。该系统已经应用到实际的GPS产品中并取得较好的效果。 关键词:地磁;SPI接口;V2XE;磁场传感器;应用 中图分类号:TN753.8 文献标识码:A 文章编号:1006—6977(2006)01—0067—03 1 引言 V2XE是一种以集成微处理器作为控制和接口的新型2轴罗盘和大地磁场传感器。V2XE使用新型系统常用的3V工作电源,具有功耗低、尺寸小、在各种条件
[传感技术]
英飞凌采用Qt图形解决方案增强Traveo T2G MCU系列,实现智能渲染技术
【2024年3月7日,德国慕尼黑讯】 在竞争激烈的全球半导体市场,制造商一直在努力缩短产品上市时间。同时,他们对流畅、高分辨率图形显示器的需求也在日益增长。 为了满足这些市场需求,英飞凌科技股份公司(FSE代码:IFX / OTCQX代码:IFNNY)宣布与科尤特(Qt Group)展开战略合作。科尤特是一家全球软件公司,为整个软件开发生命周期提供跨平台解决方案。此次合作将科尤特的轻量级、高性能图形框架加入到英飞凌拥有图形功能的TRAVEO™ T2G cluster微控制器系列,标志着图形用户界面(GUI)开发模式的转变。 TRAVEO T2G集群 如今的微控制器具有丰富的图形功能,能够实现紧凑设计、高成本效益和更低的功
[物联网]
英飞凌采用Qt图形解决方案增强Traveo T2G <font color='red'>MCU</font>系列,实现智能渲染技术
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved