《電子技術(shù)應(yīng)用》
您所在的位置:首頁(yè) > 嵌入式技術(shù) > 設(shè)計(jì)應(yīng)用 > 用C語(yǔ)言實(shí)現(xiàn)繼承的研究
用C語(yǔ)言實(shí)現(xiàn)繼承的研究
楊韜
(廣州致遠(yuǎn)電子股份有限公司,廣東 廣州 510660)
摘要: C語(yǔ)言在嵌入式軟件開(kāi)發(fā)中被廣泛使用,但由于開(kāi)發(fā)人員和應(yīng)用場(chǎng)景等原因,面向?qū)ο?、設(shè)計(jì)模式等優(yōu)秀的軟件開(kāi)發(fā)方法始終沒(méi)有很好地運(yùn)用起來(lái)。時(shí)至今日,物聯(lián)網(wǎng)等應(yīng)用的興起給嵌入式軟件開(kāi)發(fā)帶來(lái)新的挑戰(zhàn),而傳統(tǒng)的面向過(guò)程開(kāi)發(fā)方法已經(jīng)難以支撐這些復(fù)雜的應(yīng)用。因此,有必要在嵌入式軟件開(kāi)發(fā)中引入面向?qū)ο蟆⒃O(shè)計(jì)模式等優(yōu)秀的軟件開(kāi)發(fā)方法。面向?qū)ο笫乾F(xiàn)在軟件方法的根基,繼承是面向?qū)ο蟮娜筇匦灾?,本文結(jié)合C語(yǔ)言的特性,對(duì)使用C語(yǔ)言實(shí)現(xiàn)繼承進(jìn)行了討論。
Abstract:
Key words :

  楊韜

  (廣州致遠(yuǎn)電子股份有限公司,廣東 廣州 510660)

        摘要C語(yǔ)言在嵌入式軟件開(kāi)發(fā)中被廣泛使用,但由于開(kāi)發(fā)人員和應(yīng)用場(chǎng)景等原因,面向?qū)ο?/a>、設(shè)計(jì)模式等優(yōu)秀的軟件開(kāi)發(fā)方法始終沒(méi)有很好地運(yùn)用起來(lái)。時(shí)至今日,物聯(lián)網(wǎng)等應(yīng)用的興起給嵌入式軟件開(kāi)發(fā)帶來(lái)新的挑戰(zhàn),而傳統(tǒng)的面向過(guò)程開(kāi)發(fā)方法已經(jīng)難以支撐這些復(fù)雜的應(yīng)用。因此,有必要在嵌入式軟件開(kāi)發(fā)中引入面向?qū)ο蟆⒃O(shè)計(jì)模式等優(yōu)秀的軟件開(kāi)發(fā)方法。面向?qū)ο笫乾F(xiàn)在軟件方法的根基,繼承是面向?qū)ο蟮娜筇匦灾唬疚慕Y(jié)合C語(yǔ)言的特性,對(duì)使用C語(yǔ)言實(shí)現(xiàn)繼承進(jìn)行了討論。

  關(guān)鍵詞: C語(yǔ)言;面向?qū)ο螅?a class="innerlink" href="http://ihrv.cn/tags/類(lèi)" title="類(lèi)" target="_blank">類(lèi);繼承

  中圖分類(lèi)號(hào):TP312文獻(xiàn)標(biāo)識(shí)碼:ADOI: 10.19358/j.issn.1674-7720.2016.24.005

  引用格式:楊韜. 用C語(yǔ)言實(shí)現(xiàn)繼承的研究[J].微型機(jī)與應(yīng)用,2016,35(24):16-18.

  0引言

  物聯(lián)網(wǎng)等應(yīng)用的興起,給嵌入式軟件開(kāi)發(fā)帶來(lái)新的挑戰(zhàn),而傳統(tǒng)的面向過(guò)程開(kāi)發(fā)方法已經(jīng)難以支撐這些復(fù)雜的應(yīng)用。因此,有必要在嵌入式軟件開(kāi)發(fā)中引入面向?qū)ο?、設(shè)計(jì)模式等優(yōu)秀的軟件開(kāi)發(fā)方法。在C++等面向?qū)ο笳Z(yǔ)言中對(duì)類(lèi)做了原生的支持,提供了class這一數(shù)據(jù)類(lèi)型,能夠很自然地支持繼承這一面向?qū)ο筇匦?。盡管C語(yǔ)言并不支持class,但是能夠通過(guò)一些特殊的處理來(lái)模擬繼承,本文將討論如何使用C語(yǔ)言來(lái)實(shí)現(xiàn)繼承這一面向?qū)ο筇匦浴?/p>

1基本概念[1]

  1.1類(lèi)

  面向?qū)ο笥腥筇匦裕悍庋b、繼承、多態(tài),這些特性主要通過(guò)類(lèi)來(lái)體現(xiàn)。類(lèi)就是一個(gè)封裝了屬性以及相關(guān)操作的代碼的邏輯實(shí)體。

  類(lèi)具有屬性,它是對(duì)象狀態(tài)的抽象,用數(shù)據(jù)結(jié)構(gòu)來(lái)描述類(lèi)的屬性。

  類(lèi)具有方法,它是對(duì)象行為的抽象,用方法名和實(shí)現(xiàn)該操作的方法來(lái)描述。

  除了封裝屬性和操作外,類(lèi)還具有訪問(wèn)控制的能力,比如,某些屬性和方法可以是私有的,不能被外界訪問(wèn)。通過(guò)訪問(wèn)控制,能夠?qū)?nèi)部數(shù)據(jù)提供不同級(jí)別的保護(hù),以防止外界意外地改變或使用了私有部分。不同的編程語(yǔ)言提供的訪問(wèn)控制等級(jí)不盡相同,但都有公有、私有兩個(gè)等級(jí)。

  類(lèi)是抽象的數(shù)據(jù)類(lèi)型,在內(nèi)存中并不存在(Python等動(dòng)態(tài)語(yǔ)言除外),只有類(lèi)的實(shí)例存在于內(nèi)存中。

  1.2繼承

  在定義一個(gè)類(lèi)的時(shí)候,可以在一個(gè)已經(jīng)存在的類(lèi)的基礎(chǔ)上進(jìn)行,新的類(lèi)自動(dòng)繼承已存在類(lèi)的公有屬性和方法,在此基礎(chǔ)上可以添加新的屬性或方法,這種特性就是繼承。被繼承的類(lèi)稱作父類(lèi)或基類(lèi),繼承而得到的新類(lèi)稱作子類(lèi)或派生類(lèi)。通過(guò)繼承可以使開(kāi)發(fā)的軟件具有擴(kuò)展性,簡(jiǎn)化了類(lèi)的創(chuàng)建工作量,提高了代碼復(fù)用性。

 

001.jpg

  圖1為類(lèi)繼承的UML圖,圖中定義了兩個(gè)類(lèi),兩個(gè)類(lèi)用空心三角箭頭連接,箭頭指向的就是父類(lèi)Human,箭尾就是子類(lèi)Chinese。Chinese類(lèi)繼承了Human類(lèi),Chinese類(lèi)自動(dòng)擁有Human的公有屬性和方法(即name、buy()和talk()),此外,Chinese類(lèi)新添加了方法play_mahjong()。通俗點(diǎn)描述就是:中國(guó)人是人類(lèi),有名字,能夠講話和購(gòu)物,除此之外,還能打麻將。

  繼承分為單重繼承和多重繼承:子類(lèi)只繼承一個(gè)父類(lèi),稱為單重繼承,如圖1所示;子類(lèi)繼承多個(gè)父類(lèi),稱為多重繼承,如圖2所示。為了避免二義性,不推薦使用多重繼承,本文只討論單重繼承。

002.jpg

2類(lèi)的C語(yǔ)言實(shí)現(xiàn)

  在C語(yǔ)言中可以使用.c、.h和結(jié)構(gòu)體來(lái)實(shí)現(xiàn)類(lèi),以圖1中Human類(lèi)為例,可以使用human.h、human.c、struct human三個(gè)元素來(lái)完成封裝,human.c為human.h中函數(shù)聲明的實(shí)現(xiàn),本文不討論這些細(xì)節(jié),只給出human.h的關(guān)鍵代碼片段:

  程序清單1Human類(lèi)C語(yǔ)言實(shí)現(xiàn)

  // human.h

  typedef struct human {

  const char *name;

  int _money;

  } human_t;

  human_t *human_init(human_t *p_this,

  const char *name,

  int money);

  voidhuman_talk(human_t *p_this,

  const char *p_words);

  voidhuman_buy (human_t *p_this,

  const char *p_something,

  unsigned price,

  unsigned count);

  voidhuman_deinit(human_t *p_this);

3繼承的C語(yǔ)言實(shí)現(xiàn)

  3.1C語(yǔ)言不能實(shí)現(xiàn)嚴(yán)格的繼承

  一種常見(jiàn)的用C語(yǔ)言實(shí)現(xiàn)繼承的方法如下面的代碼所示:

  /* 父類(lèi) /基類(lèi)*/

  struct parent {

  int a;

  };

  /* 子類(lèi)/派生類(lèi) */

  struct child {

  struct parent base; /* 第一個(gè)成員為基類(lèi) */

  int b;

  };

  void foo (void)

  {

  struct childfoo;

  struct child *p_child;

  struct parent*p_parent;

  p_child = &foo;

  p_parent = (struct parent *)p_child;

  /* 將子類(lèi)轉(zhuǎn)換為父類(lèi) */

  p_parent->a = 100;/* 訪問(wèn)父類(lèi)成員 */

  }

  上面的代碼中定義了一個(gè)父類(lèi)和子類(lèi),foo()函數(shù)中實(shí)例化了一個(gè)子類(lèi)對(duì)象,使用強(qiáng)制類(lèi)型轉(zhuǎn)換將子類(lèi)對(duì)象的指針p_child轉(zhuǎn)換為父類(lèi)指針p_parent,如此達(dá)到了訪問(wèn)其父類(lèi)成員的效果。此方法有明顯的缺陷——使用了強(qiáng)制類(lèi)型轉(zhuǎn)換,而在C語(yǔ)言編程中是要避免使用強(qiáng)制類(lèi)型轉(zhuǎn)換的。如果要得到子類(lèi)的父類(lèi),推薦下面這種更安全的方法:

  p_parent = &p_child->base;

  對(duì)于很多面向?qū)ο缶幊陶Z(yǔ)言來(lái)說(shuō),子類(lèi)對(duì)象調(diào)用父類(lèi)的屬性方法不需要顯式轉(zhuǎn)型,而C語(yǔ)言做不到這一點(diǎn),比如,不能通過(guò)p_child->a直接訪問(wèn)父類(lèi)的屬性,因此,嚴(yán)格意義上說(shuō)“C語(yǔ)言不能實(shí)現(xiàn)嚴(yán)格的繼承”。

  3.2用C語(yǔ)言實(shí)現(xiàn)繼承

  在前面一節(jié)中指出“C語(yǔ)言不能實(shí)現(xiàn)嚴(yán)格的繼承”,盡管如此,由于繼承在軟件設(shè)計(jì)中時(shí)有使用,因此用C語(yǔ)言實(shí)現(xiàn)繼承仍是必要的。盡管繼承實(shí)現(xiàn)的效果不如C++等面向?qū)ο笳Z(yǔ)言那么完美,但還是可以達(dá)到實(shí)用程度的。

  以圖1為例,Human為父類(lèi),Chinese為基類(lèi)。Human類(lèi)的實(shí)現(xiàn)請(qǐng)參考程序清單1,Chinese類(lèi)的實(shí)現(xiàn)(chinese.h)請(qǐng)參考程序清單2,chinese.c為chinese.h中函數(shù)聲明的實(shí)現(xiàn),本文不討論這些細(xì)節(jié)。

  程序清單2Chinese類(lèi)C語(yǔ)言實(shí)現(xiàn)

  #include "human.h"

  typedef struct chinese {

  human_t super;

  const char *city;

  } chinese_t;

  #define CHINESE_TO_HUMAN(p_chinese) \  (&((p_chinese)->super))

  chinese_t *chinese_init (chinese_t *p_this, const char *name, int money, const char *city);

  chinese_t *chinese_create(const char *name, unsigned int money, const char *city);

  void chinese_play_mahjong (chinese_t *p_this);

  void chinese_deinit (chinese_t *p_this);

  void chinese_delete (chinese_t **pp_this);

  Chinese類(lèi)繼承Human類(lèi)體現(xiàn)在struct chinese 結(jié)構(gòu)體中嵌入了其父類(lèi)struct human成員,但這并不是完美的繼承,如果要訪問(wèn)父類(lèi)的屬性和方法,需要先調(diào)用CHINESE_TO_HUMAN()將子類(lèi)指針轉(zhuǎn)型為父類(lèi)指針。需要注意的是CHINESE_TO_HUMAN()并沒(méi)有使用強(qiáng)制類(lèi)型轉(zhuǎn)換,這意味著struct chinese的成員super可以放在任意位置,大大提高了使用的安全性和靈活性。程序清單3展示了繼承相關(guān)特性的使用。

  程序清單3繼承的使用

  chinese_txiaoming, *p_xiaoming;

  human_t*p_human;

  p_ xiaoming = chinese_create(

  "XiaoMing", 100, "Beijing"); // 實(shí)例化子類(lèi)

  p_human = CHINESE_TO_HUMAN(p_ xiaoming);

  // 向上轉(zhuǎn)型,得到父類(lèi)引用

  human_talk(p_human, "Ni Hao!\\n");// 調(diào)用父類(lèi)方法

  chinese_play_mahjong(p_laowang);// 調(diào)用子類(lèi)方法

4結(jié)論

  本文通過(guò)使用C語(yǔ)言實(shí)現(xiàn)Chinese類(lèi)對(duì)Human類(lèi)的繼承,討論了如何使用C語(yǔ)言來(lái)實(shí)現(xiàn)繼承。在C++等面向?qū)ο笳Z(yǔ)言中對(duì)類(lèi)做了原生的支持,能夠很容易地實(shí)現(xiàn)。盡管C語(yǔ)言不能實(shí)現(xiàn)嚴(yán)格意義上的繼承,但是通過(guò)在一個(gè)結(jié)構(gòu)體中嵌入另一個(gè)結(jié)構(gòu)體的方式,也能達(dá)到繼承的效果,與其他面向?qū)ο笳Z(yǔ)言不同的是,調(diào)用父類(lèi)方法時(shí)需要顯式轉(zhuǎn)型。

參考文獻(xiàn)

 ?。?] 百度百科. 面向?qū)ο螅跡B/OL].(2012 12 12)[2016 08 08]http://baike.baidu.com/linkurl=6XlXEOSlrKn87S7SJv4U WSX7EjstoDVmwJ13OAodXUrUrnZkVg3ntPFirEy5c6mqObZZ OevQI6K3Ungq1Mq.

  

此內(nèi)容為AET網(wǎng)站原創(chuàng),未經(jīng)授權(quán)禁止轉(zhuǎn)載。