嵌入式C實(shí)戰(zhàn)項(xiàng)目開(kāi)發(fā)技巧:對(duì)一個(gè)有規(guī)律的數(shù)組表進(jìn)行位移操作的方法
在嵌入式項(xiàng)目開(kāi)發(fā)中,LED燈的操作是一定要會(huì)的,也是基礎(chǔ)中的基礎(chǔ),比如用51單片機(jī)寫(xiě)個(gè)跑馬燈,這不簡(jiǎn)單嘛,定義一個(gè)數(shù)組把那8個(gè)跑馬燈存起來(lái),然后搞個(gè)for循環(huán)不就可以了嘛,但是,實(shí)際工作開(kāi)發(fā)中寫(xiě)一個(gè)跑馬燈可不像學(xué)校和書(shū)本上那么簡(jiǎn)單噢,往往最簡(jiǎn)單的東西,有可能也是最復(fù)雜的。現(xiàn)在我的需求是這樣的,我要求實(shí)現(xiàn)以下形式的流水燈:
跑馬燈在這個(gè)表格中是一位一位進(jìn)行存儲(chǔ)的,如果要一行全亮,那么寫(xiě)0xff,燈就全亮了,寫(xiě)0x00,燈就全滅了。

要求從led1流水到led100這個(gè)燈,也就是實(shí)現(xiàn)100個(gè)燈的跑馬燈操作。那如何實(shí)現(xiàn)呢?有人肯定會(huì)說(shuō)定義十個(gè)數(shù)組就行啦,用填表的方法。然后搞十個(gè)for循環(huán)來(lái)實(shí)現(xiàn),那么可取不?可取,但是太麻煩啦,因?yàn)檫@是用空間來(lái)?yè)Q時(shí)間,雖然能達(dá)到效果,但是就太浪費(fèi)內(nèi)存空間了,效率也是很低的。
那么如何快速處理這個(gè)問(wèn)題呢?
答案就是位運(yùn)算與循環(huán)的結(jié)合,首先思考一下,led是從第1個(gè)開(kāi)始一直流到第100個(gè),一共有十行,每行有十個(gè),那么我們就可以定義一個(gè)for循環(huán)來(lái)循環(huán)相應(yīng)的行數(shù),設(shè)定led燈的初始位置,然后用移位算法,移動(dòng)8位就換下一行,一個(gè)led相當(dāng)于1bit,這樣的話(huà),兩個(gè)for循環(huán)就可以搞定了,接下來(lái)我們用C語(yǔ)言來(lái)模擬這個(gè)過(guò)程。
#include <stdio.h>
void delay()
{
int i , j ;
for(i = 0 ; i < 1000 ; i++)
for(j = 0 ; j < 8000 ;j++);
}
//跑馬燈從高位到低位流
void test_low_to_high_bit()
{
int i , j ;
int tick ;
static int ledbuf[10] ;
for(i = 0 ; i < 10 ; i++)
{
ledbuf[i] = 0x01 ; //設(shè)定每次開(kāi)始的位置,從0x01開(kāi)始
for(j = 0 ; j < 8 ; j++)
{
printf("%p ",ledbuf[i]);
ledbuf[i] <<= 1 ; //每次左移一位,左移八次
delay() ;
}
putchar('\n');
ledbuf[i] = 0 ; //將數(shù)組清0,等待進(jìn)入下一行
}
}
//跑馬燈從低位到高位流
void test_high_to_low_bit()
{
int i , j ;
int tick ;
static int ledbuf[10] ;
for(i = 0 ; i < 10 ; i++)
{
ledbuf[i] = 0x80 ; //設(shè)定從高位的第一個(gè)LED燈的位置
for(j = 0 ; j < 8 ; j++)
{
printf("%p ",ledbuf[i]);
ledbuf[i] >>= 1 ; //將數(shù)組的第一個(gè)元素右移一位,一共右移八位
delay() ;
}
putchar('\n');
ledbuf[i] = 0 ;//將數(shù)組清0,等待進(jìn)入下一行</span>
}
}
int main(void)
{
test_low_to_high_bit() ;
putchar('\n');
test_high_to_low_bit() ;
return 0 ;
}
運(yùn)行結(jié)果:

運(yùn)行結(jié)果很明顯,第一個(gè)是從第一位移位到第八位,移動(dòng)了8次,移動(dòng)了十行。
第二個(gè)是反著來(lái)的。
如果,現(xiàn)在有個(gè)需求改了,那個(gè)表中,我要指定的LED亮,其它的不亮,然后實(shí)現(xiàn)流水燈的效果,那要如何去寫(xiě)這個(gè)程序?
這種問(wèn)題的分析就要復(fù)雜很多啦,所以,一個(gè)流水燈,簡(jiǎn)單可以簡(jiǎn)單到不用兩分鐘你可以寫(xiě)出來(lái),復(fù)雜可以復(fù)雜到可能你想一天都想不出來(lái),所以,無(wú)論遇到什么問(wèn)題,做一下總結(jié),下次遇到問(wèn)題的時(shí)候就可以熟能生巧。沒(méi)有寫(xiě)不出的能實(shí)現(xiàn)程序,沒(méi)有實(shí)現(xiàn)不了的軟件需求。只有頭腦簡(jiǎn)單的程序員。加油吧,各位!
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章
VC++實(shí)現(xiàn)文件與應(yīng)用程序關(guān)聯(lián)的方法(注冊(cè)表修改)
這篇文章主要介紹了VC++實(shí)現(xiàn)文件與應(yīng)用程序關(guān)聯(lián)的方法,涉及VC++針對(duì)注冊(cè)表的相關(guān)操作技巧,需要的朋友可以參考下2016-08-08
C語(yǔ)言中查找字符在字符串中出現(xiàn)的位置的方法
這篇文章主要介紹了C語(yǔ)言中查找字符在字符串中出現(xiàn)的位置的方法,分別是strchr()函數(shù)和strrchr()函數(shù)的使用,需要的朋友可以參考下2015-08-08
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)字符串分割的實(shí)例
這篇文章主要介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)字符串分割的實(shí)例的相關(guān)資料,希望通過(guò)本文能幫助到大家實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-10-10
C++ 設(shè)置和獲取當(dāng)前工作路徑的實(shí)現(xiàn)代碼
這篇文章主要介紹了C++ 設(shè)置和獲取當(dāng)前工作路徑的實(shí)現(xiàn)代碼,防止DLL加載不到配置和文件,需要的朋友可以參考下2017-09-09
c++實(shí)現(xiàn)MD5算法實(shí)現(xiàn)代碼
用c++實(shí)現(xiàn)了md5算法。包含 md5.h 和md5.cpp 兩個(gè)文件。主要參考百度百科 “MD5” 原理,代碼中變量命名也是參考其中的公式,程序的使用說(shuō)明在md5.h 文件的末尾注釋中2013-11-11
C++讀寫(xiě)(CSV,Yaml,二進(jìn)制)文件的方法詳解
為了處理文件,我們可以利用fstream庫(kù)。在這個(gè)庫(kù)里面有三種數(shù)據(jù)類(lèi)型:ofstream,ifstream,fstream。本文將利用這個(gè)庫(kù)實(shí)現(xiàn)不同文件的讀寫(xiě)操作,需要的可以參考一下2022-05-05
C++讀取WAV音頻文件的頭部數(shù)據(jù)的實(shí)現(xiàn)方法
這篇文章主要介紹了C++讀取WAV音頻文件的頭部數(shù)據(jù)的實(shí)現(xiàn)方法的相關(guān)資料,希望通過(guò)本文能幫助到大家,讓大家實(shí)現(xiàn)這樣的方法,需要的朋友可以參考下2017-10-10

