嵌入式C實(shí)戰(zhàn)項(xiàng)目開發(fā)技巧:對(duì)一個(gè)有規(guī)律的數(shù)組表進(jìn)行位移操作的方法
在嵌入式項(xiàng)目開發(fā)中,LED燈的操作是一定要會(huì)的,也是基礎(chǔ)中的基礎(chǔ),比如用51單片機(jī)寫個(gè)跑馬燈,這不簡(jiǎn)單嘛,定義一個(gè)數(shù)組把那8個(gè)跑馬燈存起來,然后搞個(gè)for循環(huán)不就可以了嘛,但是,實(shí)際工作開發(fā)中寫一個(gè)跑馬燈可不像學(xué)校和書本上那么簡(jiǎn)單噢,往往最簡(jiǎn)單的東西,有可能也是最復(fù)雜的。現(xiàn)在我的需求是這樣的,我要求實(shí)現(xiàn)以下形式的流水燈:
跑馬燈在這個(gè)表格中是一位一位進(jìn)行存儲(chǔ)的,如果要一行全亮,那么寫0xff,燈就全亮了,寫0x00,燈就全滅了。
要求從led1流水到led100這個(gè)燈,也就是實(shí)現(xiàn)100個(gè)燈的跑馬燈操作。那如何實(shí)現(xiàn)呢?有人肯定會(huì)說定義十個(gè)數(shù)組就行啦,用填表的方法。然后搞十個(gè)for循環(huán)來實(shí)現(xiàn),那么可取不?可取,但是太麻煩啦,因?yàn)檫@是用空間來換時(shí)間,雖然能達(dá)到效果,但是就太浪費(fèi)內(nèi)存空間了,效率也是很低的。
那么如何快速處理這個(gè)問題呢?
答案就是位運(yùn)算與循環(huán)的結(jié)合,首先思考一下,led是從第1個(gè)開始一直流到第100個(gè),一共有十行,每行有十個(gè),那么我們就可以定義一個(gè)for循環(huán)來循環(huán)相應(yīng)的行數(shù),設(shè)定led燈的初始位置,然后用移位算法,移動(dòng)8位就換下一行,一個(gè)led相當(dāng)于1bit,這樣的話,兩個(gè)for循環(huán)就可以搞定了,接下來我們用C語言來模擬這個(gè)過程。
#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è)定每次開始的位置,從0x01開始 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è)是反著來的。
如果,現(xiàn)在有個(gè)需求改了,那個(gè)表中,我要指定的LED亮,其它的不亮,然后實(shí)現(xiàn)流水燈的效果,那要如何去寫這個(gè)程序?
這種問題的分析就要復(fù)雜很多啦,所以,一個(gè)流水燈,簡(jiǎn)單可以簡(jiǎn)單到不用兩分鐘你可以寫出來,復(fù)雜可以復(fù)雜到可能你想一天都想不出來,所以,無論遇到什么問題,做一下總結(jié),下次遇到問題的時(shí)候就可以熟能生巧。沒有寫不出的能實(shí)現(xiàn)程序,沒有實(shí)現(xiàn)不了的軟件需求。只有頭腦簡(jiǎn)單的程序員。加油吧,各位!
總結(jié)
以上就是這篇文章的全部內(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-08C語言數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)字符串分割的實(shí)例
這篇文章主要介紹了C語言數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)字符串分割的實(shí)例的相關(guān)資料,希望通過本文能幫助到大家實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-10-10C++ 設(shè)置和獲取當(dāng)前工作路徑的實(shí)現(xiàn)代碼
這篇文章主要介紹了C++ 設(shè)置和獲取當(dāng)前工作路徑的實(shí)現(xiàn)代碼,防止DLL加載不到配置和文件,需要的朋友可以參考下2017-09-09c++實(shí)現(xiàn)MD5算法實(shí)現(xiàn)代碼
用c++實(shí)現(xiàn)了md5算法。包含 md5.h 和md5.cpp 兩個(gè)文件。主要參考百度百科 “MD5” 原理,代碼中變量命名也是參考其中的公式,程序的使用說明在md5.h 文件的末尾注釋中2013-11-11C++讀寫(CSV,Yaml,二進(jìn)制)文件的方法詳解
為了處理文件,我們可以利用fstream庫。在這個(gè)庫里面有三種數(shù)據(jù)類型:ofstream,ifstream,fstream。本文將利用這個(gè)庫實(shí)現(xiàn)不同文件的讀寫操作,需要的可以參考一下2022-05-05C++讀取WAV音頻文件的頭部數(shù)據(jù)的實(shí)現(xiàn)方法
這篇文章主要介紹了C++讀取WAV音頻文件的頭部數(shù)據(jù)的實(shí)現(xiàn)方法的相關(guān)資料,希望通過本文能幫助到大家,讓大家實(shí)現(xiàn)這樣的方法,需要的朋友可以參考下2017-10-10