鍍金池/ 教程/ 數(shù)據(jù)庫/ 9.7 28BYJ-48 步進(jìn)電機(jī)控制程序基礎(chǔ)
8.3 C 語言函數(shù)的形參和實(shí)參
12.2 C 語言指針變量的聲明
12.5 ?C 語言字符數(shù)組和字符指針
7.3 單片機(jī) LED 點(diǎn)陣的介紹
11.5 UART 串口通信的基本應(yīng)用
9.9 單片機(jī)蜂鳴器控制程序和驅(qū)動電路
10. 單片機(jī)實(shí)例練習(xí)與經(jīng)驗(yàn)積累
10.3 單片機(jī)交通燈控制程序和設(shè)計(jì)原理
9.8 實(shí)用的 28BYJ-48 步進(jìn)電機(jī)控制程序
8.2 C 語言函數(shù)的調(diào)用
12.4 C 語言指向數(shù)組元素的指針
7.1 C 語言變量的作用域
11.2 RS232 通信接口
12.7 1602 液晶的讀寫時(shí)序介紹
7.2 C 語言變量的存儲類別
8. C 語言函數(shù)進(jìn)階與單片機(jī)按鍵
10.4 51單片機(jī) RAM 區(qū)域的劃分
12.1 C 語言變量的地址
11. UART 串口通信
7. 變量進(jìn)階與點(diǎn)陣 LED
8.4 單片機(jī)按鍵介紹
9.3 電機(jī)的分類
9.1 單片機(jī) IO 口的結(jié)構(gòu)
單片機(jī)通信實(shí)例與 ASCII 碼
8.1 單片機(jī)最小系統(tǒng)解析(電源、晶振和復(fù)位電路)
9.2 單片機(jī)上下拉電阻
11.4 單片機(jī) IO 口模擬 UART 串口通信
9.5 讓 28BYJ-48 步進(jìn)電機(jī)轉(zhuǎn)起來
9.7 28BYJ-48 步進(jìn)電機(jī)控制程序基礎(chǔ)
12.8 1602 液晶指令介紹
12.3 C 語言指針的簡單示例
8.7 單片機(jī)矩陣按鍵的掃描
7.4 單片機(jī) LED 點(diǎn)陣的圖形顯示
8.6 單片機(jī)按鍵消抖程序
10.2 單片機(jī)中 PWM 的原理與控制程序
7.6 單片機(jī) LED 點(diǎn)陣的橫向移動(動態(tài)顯示)
11.3 USB 轉(zhuǎn)串口通信
12.9 1602 液晶簡單顯示程序
9.4 28BYJ-48 步進(jìn)電機(jī)原理
8.5 ?單片機(jī)獨(dú)立按鍵掃描程序
12. C 語言指針基礎(chǔ)與1602液晶的初步認(rèn)識
9. 單片機(jī)中的步進(jìn)電機(jī)與蜂鳴器
10.1 單片機(jī)數(shù)字秒表程序
7.5 單片機(jī) LED 點(diǎn)陣的縱向移動(動態(tài)顯示)
8.8 單片機(jī)簡易加法計(jì)算器程序
11.1 單片機(jī)串行通信介紹
10.5 單片機(jī)長短按鍵的應(yīng)用
12.6 1602 液晶介紹(電路和引腳圖)
9.6 28BYJ-48 步進(jìn)電機(jī)轉(zhuǎn)動精度與深入分析

9.7 28BYJ-48 步進(jìn)電機(jī)控制程序基礎(chǔ)

解決了精度問題,讓我們再次回到我們的電機(jī)控制程序上吧。上面給出的兩個(gè)例程都不是實(shí)用的程序,為什么?因?yàn)槌绦蛑写嬖诖蠖蔚难訒r(shí),而在延時(shí)的時(shí)候是什么其它的事都干不了的,想想第二個(gè)程序,整整200秒什么別的事都干不了,這在實(shí)際的控制系統(tǒng)中是絕對不允許的。那么怎么改造一下呢?當(dāng)然還是用定時(shí)中斷來完成了,既然每個(gè)節(jié)拍持續(xù)時(shí)間是 2 ms,那我們直接用定時(shí)器定時(shí) 2 ms 來刷新節(jié)拍就行了。改造后的程序如下:

#include <reg52.h>
unsigned long beats = 0; //電機(jī)轉(zhuǎn)動節(jié)拍總數(shù)
void StartMotor(unsigned long angle);
void main(){
    EA = 1;  //使能總中斷
    TMOD = 0x01; //設(shè)置 T0 為模式1
    TH0 = 0xF8; //為 T0 賦初值 0xF8CD,定時(shí) 2 ms
    TL0 = 0xCD;
    ET0 = 1; //使能 T0 中斷
    TR0 = 1; //啟動 T0
    StartMotor(360*2+180); //控制電機(jī)轉(zhuǎn)動2圈半
    while (1);
}
/* 步進(jìn)電機(jī)啟動函數(shù),angle-需轉(zhuǎn)過的角度 */
void StartMotor(unsigned long angle){
    //在計(jì)算前關(guān)閉中斷,完成后再打開,以避免中斷打斷計(jì)算過程而造成錯(cuò)誤
    EA = 0;
    beats = (angle * 4076) / 360; //實(shí)測為4076拍轉(zhuǎn)動一圈
    EA = 1;
}
/* T0 中斷服務(wù)函數(shù),用于驅(qū)動步進(jìn)電機(jī)旋轉(zhuǎn) */
void InterruptTimer0() interrupt 1{
    unsigned char tmp;  //臨時(shí)變量
    static unsigned char index = 0; //節(jié)拍輸出索引
    unsigned char code BeatCode[8] = { //步進(jìn)電機(jī)節(jié)拍對應(yīng)的 IO 控制代碼
        0xE, 0xC, 0xD, 0x9, 0xB, 0x3, 0x7, 0x6
    };

    TH0 = 0xF8; //重新加載初值
    TL0 = 0xCD;
    //節(jié)拍數(shù)不為 0 則產(chǎn)生一個(gè)驅(qū)動節(jié)拍
    if (beats != 0){
        tmp = P1; //用 tmp 把 P1 口當(dāng)前值暫存
        tmp = tmp & 0xF0; //用&操作清零低4位
        //用|操作把節(jié)拍代碼寫到低4位
        tmp = tmp | BeatCode[index];
        //把低4位的節(jié)拍代碼和高4位的原值送回 P1
        P1 = tmp;
        index++; //節(jié)拍輸出索引遞增
        index = index & 0x07; //用&操作實(shí)現(xiàn)到8歸零
        beats--; //總節(jié)拍數(shù)-1
    }else{ //節(jié)拍數(shù)為0則關(guān)閉電機(jī)所有的相
        P1 = P1 | 0x0F;
    }
}

程序還是比較簡單的,電機(jī)轉(zhuǎn)動的啟動函數(shù) StartMotor 只負(fù)責(zé)計(jì)算一個(gè)需要的總節(jié)拍數(shù) beats,然后在中斷函數(shù)內(nèi)檢測這個(gè)變量,不為0時(shí)就執(zhí)行節(jié)拍操作,同時(shí)將其減1,直到減到0為止。

這里,我們要特別說明一下的是 StartMotor 函數(shù)中對 EA 的兩次操作。我們可以看到對 beats 的賦值計(jì)算語句是夾在 EA=0;EA=1;這兩行語句中間的,也就是說這行賦值計(jì)算語句在執(zhí)行前先關(guān)閉了中斷,而等它執(zhí)行完后,才又重新打開了中斷。在它執(zhí)行過程中單片機(jī)是不會響應(yīng)中斷的,即中斷函數(shù) InterruptTimer0 不會被執(zhí)行,即使這時(shí)候定時(shí)器溢出了,中斷發(fā)生了,也只能等待 EA 重新置1后,才能得到響應(yīng),中斷函數(shù) InterruptTimer0 才會被執(zhí)行。

那么為什么要這么做呢?我們來想一下:在本書開始我們就曾提到,我們所使用的 STC89C52 單片機(jī)是8位單片機(jī),這個(gè)8位的概念就是說單片機(jī)操作數(shù)據(jù)時(shí)都是按8位即按1個(gè)字節(jié)進(jìn)行的,那么要操作多個(gè)字節(jié)(不論是讀還是寫)就必須分多次進(jìn)行了。而我們程序中定義的 beats 這個(gè)變量是 unsigned long 型,它要占用4個(gè)字節(jié),那么對它的賦值最少也要分4次才能完成了。我們想象一下,假如在完成了其中第一個(gè)字節(jié)的賦值后,恰好中斷發(fā)生了,InterruptTimer0 函數(shù)得到執(zhí)行,而這個(gè)函數(shù)內(nèi)可能會對 beats 進(jìn)行減1的操作,減法就有可能發(fā)生借位,借位就會改變其它的字節(jié),但因?yàn)榇藭r(shí)其它的字節(jié)還沒有被賦入新值,于是錯(cuò)誤就會發(fā)生了,減1所得到的結(jié)果就不是預(yù)期的值了!所以要避免這種錯(cuò)誤的發(fā)生就得先暫時(shí)關(guān)閉中斷,等賦值完成后再打開中斷。而如果我們使用的是 char 或 bit 型變量的話,因?yàn)樗鼈兌际窃?CPU 的一次操作中就完成的,所以即使不關(guān)中斷,也不會發(fā)生錯(cuò)誤。問題分析清楚了,如何取舍還得根據(jù)實(shí)際情況來,遇上這類問題的時(shí)候多多考慮考慮吧。