久久精品在这里_成人99免费视频_国产激情视频一区二区在线观看_国产伦精品一区二区三区免费 _亚洲午夜免费福利视频_色狠狠色狠狠综合_av在线综合网_91毛片在线观看_欧美视频一区二区在线观看_极品美女销魂一区二区三区免费_国产亚洲欧美激情_在线免费观看不卡av_日韩不卡一区二区三区_91精品国产麻豆国产自产在线_亚洲国产精品一区二区久久恐怖片_a4yy欧美一区二区三区


曙海教育集團(tuán)論壇單片機(jī)專(zhuān)區(qū)單片機(jī)高級(jí) → 從單片機(jī)初學(xué)者邁向單片機(jī)工程師”之LED主題討論周第三章----模塊化編程...


  共有8244人關(guān)注過(guò)本帖樹(shù)形打印

主題:從單片機(jī)初學(xué)者邁向單片機(jī)工程師”之LED主題討論周第三章----模塊化編程...

美女呀,離線(xiàn),留言給我吧!
wangxinxin
  1樓 個(gè)性首頁(yè) | 博客 | 信息 | 搜索 | 郵箱 | 主頁(yè) | UC


加好友 發(fā)短信
等級(jí):青蜂俠 帖子:1393 積分:14038 威望:0 精華:0 注冊(cè):2010-11-12 11:08:23
從單片機(jī)初學(xué)者邁向單片機(jī)工程師”之LED主題討論周第三章----模塊化編程...  發(fā)帖心情 Post By:2010-12-8 10:36:00

通過(guò)上一章的學(xué)習(xí),我想你已經(jīng)掌握了如何在程序中釋放CPU了。希望能夠繼續(xù)堅(jiān)持下去。一個(gè)良好的開(kāi)始是成功的一半。我們今天所做的一切都是為了在單片機(jī)編程上做的更好。
在談?wù)摻裉斓闹黝}之前,先說(shuō)下我以前的一些經(jīng)歷。在剛開(kāi)始接觸到C語(yǔ)言程序的時(shí)候,由于學(xué)習(xí)內(nèi)容所限,寫(xiě)的程序都不是很大,一般也就幾百行而矣。所以所有的程序都完成在一個(gè)源文件里面。記得那時(shí)候大一參加學(xué)校里的一個(gè)電子設(shè)計(jì)大賽,調(diào)試了一個(gè)多星期,所有程序加起來(lái)大概將近1000行,長(zhǎng)長(zhǎng)的一個(gè)文件,從上瀏覽下來(lái)都要好半天。出了錯(cuò)誤簡(jiǎn)單的語(yǔ)法錯(cuò)誤還好定位,其它一些錯(cuò)誤,往往找半天才找的到。那個(gè)時(shí)候開(kāi)始知道了模塊化編程這個(gè)東西,也嘗試著開(kāi)始把程序分模塊編寫(xiě)。最開(kāi)始是把相同功能的一些函數(shù)(譬如1602液晶的驅(qū)動(dòng))全部寫(xiě)在一個(gè)頭文件(.h)文件里面,然后需要調(diào)用的地方包含進(jìn)去,但是很快發(fā)現(xiàn)這種方法有其局限性,很容易犯重復(fù)包含的錯(cuò)誤。
而且調(diào)用起來(lái)也很不方便。很快暑假的電子設(shè)計(jì)大賽來(lái)臨了,學(xué)校對(duì)我們的單片機(jī)軟件編程進(jìn)行了一些培訓(xùn)。由于學(xué)校歷年來(lái)參加國(guó)賽和省賽,因此積累了一定數(shù)量的驅(qū)動(dòng)模塊,那些日子,老師每天都會(huì)布置一定量的任務(wù),讓我們用這些模塊組合起來(lái),完成一定功能。而正是那些日子模塊化編程的培訓(xùn),使我對(duì)于模塊化編程有了更進(jìn)一步的認(rèn)識(shí)。并且程序規(guī)范也開(kāi)始慢慢注意起來(lái)。此后的日子,無(wú)論程序的大小,均采用模塊化編程的方式去編寫(xiě)。很長(zhǎng)一段時(shí)間以來(lái),一直有單片機(jī)愛(ài)好者在QQ上和我一起交流。有時(shí)候,他們會(huì)發(fā)過(guò)來(lái)一些有問(wèn)題的程序源文件,讓我?guī)兔π薷囊幌隆M瑯邮情L(zhǎng)長(zhǎng)的一個(gè)文件,而且命名極不規(guī)范,從頭看下來(lái),著實(shí)是痛苦,說(shuō)實(shí)話(huà),還真不如我重新給他們寫(xiě)一個(gè)更快一些,此話(huà)到不假,因?yàn)槭诸^積累了一定量的模塊,在完成一個(gè)新的系統(tǒng)時(shí)候,只需要根據(jù)上層功能需求,在底層模塊的支持下,可以很快方便的完成。而不需要從頭到尾再一磚一瓦的重新編寫(xiě)。藉此,也可以看出模塊化編程的一個(gè)好處,就是可重復(fù)利用率高。下面讓我們揭開(kāi)模塊化神秘面紗,一窺其真面目。
    C語(yǔ)言源文件 *.c
        提到C語(yǔ)言源文件,大家都不會(huì)陌生。因?yàn)槲覀兤匠?xiě)的程序代碼幾乎都在這個(gè)XX.C文件里面。編譯器也是以此文件來(lái)進(jìn)行編譯并生成相應(yīng)的目標(biāo)文件。作為模塊化編程的組成基礎(chǔ),我們所要實(shí)現(xiàn)的所有功能的源代碼均在這個(gè)文件里。理想的模塊化應(yīng)該可以看成是一個(gè)黑盒子。即我們只關(guān)心模塊提供的功能,而不管模塊內(nèi)部的實(shí)現(xiàn)細(xì)節(jié)。好比我們買(mǎi)了一部手機(jī),我們只需要會(huì)用手機(jī)提供的功能即可,不需要知曉它是如何把短信發(fā)出去的,如何響應(yīng)我們按鍵的輸入,這些過(guò)程對(duì)我們用戶(hù)而言,就是是一個(gè)黑盒子。
在大規(guī)模程序開(kāi)發(fā)中,一個(gè)程序由很多個(gè)模塊組成,很可能,這些模塊的編寫(xiě)任務(wù)被分配到不同的人。而你在編寫(xiě)這個(gè)模塊的時(shí)候很可能就需要利用到別人寫(xiě)好的模塊的借口,這個(gè)時(shí)候我們關(guān)心的是,它的模塊實(shí)現(xiàn)了什么樣的接口,我該如何去調(diào)用,至于模塊內(nèi)部是如何組織的,對(duì)于我而言,無(wú)需過(guò)多關(guān)注。而追求接口的單一性,把不需要的細(xì)節(jié)盡可能對(duì)外部屏蔽起來(lái),正是我們所需要注意的地方。
    C語(yǔ)言頭文件 *.h
        談及到模塊化編程,必然會(huì)涉及到多文件編譯,也就是工程編譯。在這樣的一個(gè)系統(tǒng)中,往往會(huì)有多個(gè)C文件,而且每個(gè)C文件的作用不盡相同。在我們的C文件中,由于需要對(duì)外提供接口,因此必須有一些函數(shù)或者是變量提供給外部其它文件進(jìn)行調(diào)用。
假設(shè)我們有一個(gè)LCD.C文件,其提供最基本的LCD的驅(qū)動(dòng)函數(shù)
    LcdPutChar(char cNewValue) ; //在當(dāng)前位置輸出一個(gè)字符
而在我們的另外一個(gè)文件中需要調(diào)用此函數(shù),那么我們?cè)撊绾巫瞿兀?br/>    頭文件的作用正是在此。可以稱(chēng)其為一份接口描述文件。其文件內(nèi)部不應(yīng)該包含任何實(shí)質(zhì)性的函數(shù)代碼。我們可以把這個(gè)頭文件理解成為一份說(shuō)明書(shū),說(shuō)明的內(nèi)容就是我們的模塊對(duì)外提供的接口函數(shù)或者是接口變量。同時(shí)該文件也包含了一些很重要的宏定義以及一些結(jié)構(gòu)體的信息,離開(kāi)了這些信息,很可能就無(wú)法正常使用接口函數(shù)或者是接口變量。但是總的原則是:不該讓外界知道的信息就不應(yīng)該出現(xiàn)在頭文件里,而外界調(diào)用模塊內(nèi)接口函數(shù)或者是接口變量所必須的信息就一定要出現(xiàn)在頭文件里,否則,外界就無(wú)法正確的調(diào)用我們提供的接口功能。因而為了讓外部函數(shù)或者文件調(diào)用我們提供的接口功能,就必須包含我們提供的這個(gè)接口描述文件----即頭文件。同時(shí),我們自身模塊也需要包含這份模塊頭文件(因?yàn)槠浒四K源文件中所需要的宏定義或者是結(jié)構(gòu)體),好比我們平常所用的文件都是一式三份一樣,模塊本身也需要包含這個(gè)頭文件。
下面我們來(lái)定義這個(gè)頭文件,一般來(lái)說(shuō),頭文件的名字應(yīng)該與源文件的名字保持一致,這樣我們便可以清晰的知道哪個(gè)頭文件是哪個(gè)源文件的描述。
        于是便得到了LCD.C的頭文件LCD.h 其內(nèi)容如下。
        #ifndef    _LCD_H_
         #define     _LCD_H_
         extern   LcdPutChar(char cNewValue) ;
        #endif

    這與我們?cè)谠次募卸x函數(shù)時(shí)有點(diǎn)類(lèi)似。不同的是,在其前面添加了extern 修飾符表明其是一個(gè)外部函數(shù),可以被外部其它模塊進(jìn)行調(diào)用。
       #ifndef     _LCD_H_
         #define     _LCD_H_
         #endif

              這個(gè)幾條條件編譯和宏定義是為了防止重復(fù)包含。假如有兩個(gè)不同源文件需要調(diào)用LcdPutChar(char cNewValue)這個(gè)函數(shù),他們分別都通過(guò)#include “Lcd.h”把這個(gè)頭文件包含了進(jìn)去。在第一個(gè)源文件進(jìn)行編譯時(shí)候,由于沒(méi)有定義過(guò) _LCD_H_ 因此 #ifndef _LCD_H_ 條件成立,于是定義_LCD_H_ 并將下面的聲明包含進(jìn)去。在第二個(gè)文件編譯時(shí)候,由于第一個(gè)文件包含時(shí)候,已經(jīng)將_LCD_H_定義過(guò)了。因此#ifndef _LCD_H_ 不成立,整個(gè)頭文件內(nèi)容就沒(méi)有被包含。假設(shè)沒(méi)有這樣的條件編譯語(yǔ)句,那么兩個(gè)文件都包含了extern LcdPutChar(char cNewValue) ; 就會(huì)引起重復(fù)包含的錯(cuò)誤。
    不得不說(shuō)的typedef
          很多朋友似乎了習(xí)慣程序中利用如下語(yǔ)句來(lái)對(duì)數(shù)據(jù)類(lèi)型進(jìn)行定義
   #define uint unsigned int
    #define uchar unsigned char

    然后在定義變量的時(shí)候 直接這樣使用
uint g_nTimeCounter = 0 ;
    不可否認(rèn),這樣確實(shí)很方便,而且對(duì)于移植起來(lái)也有一定的方便性。但是考慮下面這種情況你還會(huì) 這么認(rèn)為嗎?
#define PINT unsigned int * //定義unsigned int 指針類(lèi)型
PINT g_npTimeCounter, g_npTimeState ;

      那么你到底是定義了兩個(gè)unsigned int 型的指針變量,還是一個(gè)指針變量,一個(gè)整形變量呢?而你的初衷又是什么呢,想定義兩個(gè)unsigned int 型的指針變量嗎?如果是這樣,那么估計(jì)過(guò)不久就會(huì)到處抓狂找錯(cuò)誤了。
    慶幸的是C語(yǔ)言已經(jīng)為我們考慮到了這一點(diǎn)。typedef 正是為此而生。為了給變量起一個(gè)別名我們可以用如下的語(yǔ)句
    typedef unsigned int    uint16 ;    //給指向無(wú)符號(hào)整形變量起一個(gè)別名 uint16
      typedef unsigned int * puint16 ; //給指向無(wú)符號(hào)整形變量指針起一個(gè)別名 puint16

    在我們定義變量時(shí)候便可以這樣定義了:

uint16    g_nTimeCounter = 0 ; //定義一個(gè)無(wú)符號(hào)的整形變量
puint16 g_npTimeCounter ;    //定義一個(gè)無(wú)符號(hào)的整形變量的指針

在我們使用51單片機(jī)的C語(yǔ)言編程的時(shí)候,整形變量的范圍是16位,而在基于32的微處理下的整形變量是32位。倘若我們?cè)?位單片機(jī)下編寫(xiě)的一些代碼想要移植到32位的處理器上,那么很可能我們就需要在源文件中到處修改變量的類(lèi)型定義。這是一件龐大的工作,為了考慮程序的可移植性,在一開(kāi)始,我們就應(yīng)該養(yǎng)成良好的習(xí)慣,用變量的別名進(jìn)行定義。
如在8位單片機(jī)的平臺(tái)下,有如下一個(gè)變量定義
    uint16    g_nTimeCounter = 0 ;
        如果移植32單片機(jī)的平臺(tái)下,想要其的范圍依舊為16位。
    可以直接修改uint16 的定義,即
    typedef unsigned short int    uint16 ;
        這樣就可以了,而不需要到源文件處處尋找并修改。

將常用的數(shù)據(jù)類(lèi)型全部采用此種方法定義,形成一個(gè)頭文件,便于我們以后編程直接調(diào)用。
文件名 MacroAndConst.h
其內(nèi)容如下:
#ifndef   _MACRO_AND_CONST_H_
#define   _MACRO_AND_CONST_H_

typedef    unsigned int    uint16;
typedef    unsigned int   UINT;
typedef    unsigned int   uint;
typedef    unsigned int   UINT16;
typedef    unsigned int   WORD;
typedef    unsigned int   word;
typedef      int        int16;
typedef      int        INT16;
typedef    unsigned long uint32;

typedef    unsigned long     UINT32;
typedef    unsigned long    DWORD;
typedef    unsigned long    dword;
typedef    long            int32;
typedef    long            INT32;
typedef    signed char     int8;
typedef    signed char     INT8;
typedef    unsigned char      byte;
typedef    unsigned char     BYTE;
typedef    unsigned char     uchar;
typedef    unsigned char     UINT8;
typedef    unsigned char    uint8;
typedef    unsigned char    BOOL;

#endif

至此,似乎我們對(duì)于源文件和頭文件的分工以及模塊化編程有那么一點(diǎn)概念了。那么讓我們趁熱打鐵,將上一章的我們編寫(xiě)的LED閃爍函數(shù)進(jìn)行模塊劃分并重新組織進(jìn)行編譯。

在上一章中我們主要完成的功能是P0口所驅(qū)動(dòng)的LED以1Hz的頻率閃爍。其中用到了定時(shí)器,以及LED驅(qū)動(dòng)模塊。因而我們可以簡(jiǎn)單的將整個(gè)工程分成三個(gè)模塊,定時(shí)器模塊,LED模塊,以及主函數(shù)
對(duì)應(yīng)的文件關(guān)系如下

main.c
Timer.c --?Timer.h
Led.c      --?Led.h
在開(kāi)始重新編寫(xiě)我們的程序之前,先給大家講一下如何在KEIL中建立工程模板吧,這個(gè)模板是我一直沿用至今。希望能夠給大家一點(diǎn)啟發(fā)。
下面的內(nèi)容就主要以圖片為主了。同時(shí)輔以少量文字說(shuō)明。
我們以芯片AT89S52為例。


支持(0中立(0反對(duì)(0單帖管理 | 引用 | 回復(fù) 回到頂部

返回版面帖子列表

從單片機(jī)初學(xué)者邁向單片機(jī)工程師”之LED主題討論周第三章----模塊化編程...








簽名
久久精品在这里_成人99免费视频_国产激情视频一区二区在线观看_国产伦精品一区二区三区免费 _亚洲午夜免费福利视频_色狠狠色狠狠综合_av在线综合网_91毛片在线观看_欧美视频一区二区在线观看_极品美女销魂一区二区三区免费_国产亚洲欧美激情_在线免费观看不卡av_日韩不卡一区二区三区_91精品国产麻豆国产自产在线_亚洲国产精品一区二区久久恐怖片_a4yy欧美一区二区三区
亚洲人成精品久久久久| 欧美精品色一区二区三区| 国产精品一区二区不卡| 国产成人亚洲综合色影视| 成人av在线一区二区| 114国产精品久久免费观看| 91精品国产一区二区三区动漫 | 久久久久久久免费视频了| 欧美韩日一区二区三区四区| 亚洲三级免费电影| 青青草国产精品亚洲专区无| 国产精品69毛片高清亚洲| 7777奇米亚洲综合久久| 日本高清久久一区二区三区| 欧美羞羞免费网站| 久久综合狠狠综合久久综合88 | 亚洲妇熟xx妇色黄| 欧美a级一区二区| av成人免费在线观看| 久久视频在线观看中文字幕| 欧美视频三区在线播放| 日本一区二区三区dvd视频在线| 亚洲成年人网站在线观看| 国产麻豆视频一区| 精品国产综合区久久久久久| 在线精品视频免费播放| 久久婷婷国产综合国色天香| 亚洲国产美国国产综合一区二区| 国产美女在线观看一区| 久久综合一区| 日韩精品一区二区在线观看| 亚洲综合在线免费观看| 盗摄精品av一区二区三区| 欧美亚州在线观看| 欧美一区日本一区韩国一区| 亚洲人快播电影网| 国产精品99久久久| 国产精品欧美精品| 国产一区二区三区黄视频| 黄色91av| 精品av久久707| 青青草国产精品亚洲专区无| 97久久人人超碰caoprom欧美| 欧美自拍偷拍午夜视频| 亚洲欧洲美洲综合色网| 国产乱一区二区| 日韩亚洲不卡在线| 国产欧美精品一区| 国产精品99久久久| 亚洲一区二区在| 亚洲图片另类小说| 97免费高清电视剧观看| 欧美一区二区黄色| 麻豆91精品视频| 日韩免费中文专区| 中文子幕无线码一区tr| 国产成人在线观看| 色一情一伦一子一伦一区| 自拍偷拍欧美激情| 91女神在线视频| 欧美肥妇毛茸茸| 美女精品自拍一二三四| 天天综合狠狠精品| 中文字幕一区二区三区在线不卡 | 一区二区三区电影| 亚洲同性gay激情无套| 成人一道本在线| 欧美猛男超大videosgay| 亚洲国产精品精华液网站| 国产精品一区二区三区免费观看| 精品国产伦一区二区三区免费 | 麻豆传媒一区| 欧美国产在线观看| 97久久人人超碰| 欧美成人精品福利| 国产成人免费视频网站高清观看视频| 在线观看日本一区| 午夜精品久久久久影视| 久久99精品久久久久子伦| 国产清纯美女被跳蛋高潮一区二区久久w| 国产精品一区二区在线观看不卡| 色综合久久99| 亚洲成av人片一区二区梦乃| 久久国产日韩欧美| 日本一区二区成人| 91免费版黄色| 中文在线资源观看网站视频免费不卡| 99国产精品久| 2023国产精品自拍| 91在线观看污| 国产三级精品视频| 国产精品一区二区av| 国产日韩欧美综合一区| 91精品国产综合久久久久久丝袜 | 国产福利电影一区二区三区| 欧美午夜精品久久久久久超碰 | 国产aⅴ精品一区二区三区黄| 久久久久久久久久久久久女国产乱 | 樱花草国产18久久久久| 蜜桃麻豆www久久国产精品| 国产精品久久久99| 欧美大陆一区二区| 亚洲一区二区三区四区中文字幕| 欧美国产一二三区| 亚洲影视在线播放| 日日夜夜精品网站| 美腿丝袜亚洲一区| 91超碰这里只有精品国产| 国产一区二区三区免费在线观看| 日韩女优视频免费观看| 91丨九色丨尤物| 国产精品久久久久久久久图文区 | 欧美天堂亚洲电影院在线播放| 免费在线观看日韩欧美| 精品视频色一区| www.日韩av| 亚洲色图欧洲色图婷婷| 婷婷五月色综合| 国产精品正在播放| 国产欧美日韩精品一区| 久久青青草原一区二区| 日本中文字幕一区二区视频| 欧美一级一区二区| 精品国产乱码久久久久久蜜柚 | 日韩电影在线免费看| 欧美一区二区私人影院日本| 99国精产品一二二线| 洋洋av久久久久久久一区| 欧美日韩一区在线观看| 91视频婷婷| 午夜欧美视频在线观看| 日韩女优电影在线观看| 精品一区二区三区视频日产| 美女在线视频一区| 久久久午夜精品理论片中文字幕| 欧美中日韩免费视频| 国产精品自在在线| 日韩毛片在线免费观看| 欧美日韩一卡二卡三卡 | 欧美不卡在线一区二区三区| 美女久久久精品| 欧美极品xxx| 色婷婷综合久久久中文一区二区| av电影天堂一区二区在线观看| 亚洲精品精品亚洲| 欧美肥妇free| 欧美最大成人综合网| 国产成人午夜精品影院观看视频 | 一本一本久久a久久精品综合麻豆 一本一道波多野结衣一区二区 | 黄色日韩三级电影| 一色桃子久久精品亚洲| 欧美精品高清视频| 日本成人黄色免费看| 99精品视频在线免费观看| 日韩国产成人精品| 国产精品夫妻自拍| 日韩欧美一二区| 色综合久久中文综合久久牛| 国产传媒一区二区| 狠狠网亚洲精品| 亚洲午夜精品网| 中文字幕免费不卡| 日韩一区二区三免费高清| 在线播放豆国产99亚洲| 国产精品一区而去| 成人的网站免费观看| 奇米精品一区二区三区四区| 亚洲三级在线播放| 久久久久久久久久美女| 欧美精品一卡两卡| 色综合天天综合色综合av | 欧美精品一区二区不卡 | 亚洲乱码日产精品bd| 久久久久久久久伊人| 欧美一区二区三区在线| 在线视频亚洲自拍| 欧美中文娱乐网| 精品久久精品久久| 999视频在线免费观看| 成人免费看黄yyy456| 久久99国产精品免费| 日韩电影在线一区二区| 亚洲欧美激情插| 国产精品久久久久久妇女6080 | 国产a一区二区| 成人国产精品免费| 精品无人码麻豆乱码1区2区| 青青草视频一区| 日本中文字幕一区二区视频| 亚洲图片一区二区| 亚洲美女精品一区| 亚洲欧洲国产日本综合| 国产免费成人在线视频| 久久精品日韩一区二区三区| 亚洲精品一区二区三区蜜桃下载| 欧美一区二区播放| 日韩一区二区电影网| 777奇米成人网| 91精品国产91久久久久久一区二区 | 色偷偷久久人人79超碰人人澡|