0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何把C++的源程序改寫成C語(yǔ)言

j4AI_wujianying ? 來(lái)源:?jiǎn)纹瑱C(jī)精講吳鑒鷹 ? 2020-05-14 10:08 ? 次閱讀

曾經(jīng)參與過(guò)公司的bpp項(xiàng)目,就是bluetooth print profile。由于使用了hpijs的開(kāi)源包,但是是C++的。由于C++解釋器比C語(yǔ)言解釋器占用的存儲(chǔ)空間要大500k左右。為了節(jié)省有限的存儲(chǔ)空間,降低成本,同時(shí)也為了提高效率,將用C++語(yǔ)言寫的源程序用C語(yǔ)言改寫是很有必要的。

C++與C區(qū)別最大的就是C++中的類的概念和特性,將C++改為C的問(wèn)題,就轉(zhuǎn)換成如何將類化去的問(wèn)題。

方法有兩種:

第一種是將C++中的面向?qū)ο筇卣魅サ?,先全部理解源代碼的邏輯,然后改寫;第二種是在C中保留面向?qū)ο蟮牟糠痔卣?,用結(jié)構(gòu)體實(shí)現(xiàn)類的功能。

第一種方法,對(duì)于類的數(shù)目很少的情況還可以,如果類的數(shù)目比較多,全部理解源代碼,然后重寫就很耗時(shí)間,而且很容易出錯(cuò),更甚者,如果遇到大的項(xiàng)目想全部理解源代碼幾乎是不可能的。

hpijs程序中類有140多個(gè),這個(gè)時(shí)候就需要采用第二個(gè)方法了,你可以一個(gè)類一個(gè)類的改沒(méi)有什么太高的難度,如果不是筆誤的話,幾乎不會(huì)出錯(cuò),而且根本不需要理解程序邏輯,也許改完后你對(duì)程序所要實(shí)現(xiàn)的功能還一無(wú)所知。倒不是說(shuō)一無(wú)所知對(duì)大家有好處,只是想說(shuō)這種方法的與程序邏輯本身的無(wú)關(guān)性。

下面對(duì)C++的一些特性,以及如何在c里實(shí)現(xiàn)或者替代,作一些初步的探討:

說(shuō)明:

函數(shù)Ixx為類xx的構(gòu)造函數(shù)的實(shí)現(xiàn)。

原類的成員函數(shù)改為前綴為結(jié)構(gòu)體名+‘_’的函數(shù)。

函數(shù)指針U為原類的析構(gòu)函數(shù)的聲明;

U+結(jié)構(gòu)體名稱為原類的析構(gòu)函數(shù)的實(shí)現(xiàn);

Fun-_+結(jié)構(gòu)體名為對(duì)該結(jié)構(gòu)體成員函數(shù)指針進(jìn)行指向;

以后遇到上述情況將不再說(shuō)明。

一.類的成員函數(shù)和數(shù)據(jù)成員

由于struct沒(méi)有對(duì)成員的訪問(wèn)權(quán)限進(jìn)行控制,必須加入額外的機(jī)制進(jìn)行訪問(wèn)控制,這樣一來(lái)就使得程序復(fù)雜化了,所以只能放棄訪問(wèn)權(quán)限的控制。

1)對(duì)于類的數(shù)據(jù)成員可以直接轉(zhuǎn)為C中結(jié)構(gòu)體的數(shù)據(jù)成員。

2)函數(shù)則需轉(zhuǎn)化為對(duì)應(yīng)的函數(shù)指針,因?yàn)閟truct里不允許出現(xiàn)函數(shù)的聲明和定義。而函數(shù)前如果有virture,inline等修飾符也要去掉,如函數(shù)void funca(int a);改為void (*funca)(struct B *p,int a);大家可以看到函數(shù)指針的原型里加了一個(gè)指針struct B的指針,這是因?yàn)橐诤瘮?shù)內(nèi)部對(duì)類的成員進(jìn)行操作,要靠該指針指定結(jié)構(gòu)體的成員。在類的成員函數(shù)里,實(shí)際上在參數(shù)列里也隱含有一個(gè)指向自身的this指針。

3)對(duì)于靜態(tài)成員則要定義成全局變量或全局函數(shù),因?yàn)榻Y(jié)構(gòu)體中不能有靜態(tài)成員。

二.類的構(gòu)造函數(shù)

類在實(shí)例化的時(shí)候會(huì)調(diào)用類的缺省構(gòu)造函數(shù),在struct里,要定義一個(gè)同名函數(shù)指針指向一個(gè)具有構(gòu)造函數(shù)功能的初始化函數(shù),與構(gòu)造函數(shù)不同的是,要在初始化函數(shù)里加入進(jìn)行函數(shù)指針初始化的語(yǔ)句.使用的時(shí)候在創(chuàng)建結(jié)構(gòu)體變量的時(shí)候要用malloc而不是new,并且這個(gè)時(shí)候要手工調(diào)用初始化函數(shù)。

如下例所示:

class A{public: A(); ~A(); void func(int a);private: int b;};A::A(){ b=0;} void A::func(int a){ b=a;} typedef struct classA A;struct classA{ void (*A)(struct classA *p);//構(gòu)造函數(shù)指針 void (*U)(struct classA *p);//析構(gòu)函數(shù)指針 void (*func)(struct classA *p,int a); int b;}; void fun_A(A *p){ p->func=classA_func; //將函數(shù)指針初始化} void IA(A *p) //構(gòu)造函數(shù),命名規(guī)則在類名前加I{ fun_A(p); p->b=0; //原構(gòu)造函數(shù)所作部分} void classA_func(A *p,int a){ p->b=a;}
在使用的地方采用如下方式:

A *s=(A*)malloc(sizeof(A)); s->A=IA; s->A(s);

三.類的析構(gòu)函數(shù)


類的析構(gòu)函數(shù)所作的工作是釋放所占的資源。

在C中,無(wú)論是哪個(gè)struct都用函數(shù)指針U替代析構(gòu)函數(shù)。之所以所有的struct都用指針U是基于如下情況:

如果將子類指針賦給基類指針,基類指針在釋放的時(shí)候不必考慮調(diào)用哪個(gè)函數(shù)名的析構(gòu)函數(shù),只需調(diào)用成員函數(shù)U即可。成員函數(shù)U需要像一般成員函數(shù)一樣在fun_類名()函數(shù)中指定。

類的析構(gòu)函數(shù)是由系統(tǒng)調(diào)用的,在C中則要顯式調(diào)用。至于何時(shí)調(diào)用,要準(zhǔn)確判斷。

四.類的拷貝構(gòu)造函數(shù)


類的拷貝構(gòu)造函數(shù)主要用途是加快以下情況下類的構(gòu)建速度:

1. 作為參數(shù)傳給函數(shù)。(additem(Itema))
2. 作為函數(shù)返回值。
3. 實(shí)例化類時(shí)作參數(shù)。

這三種情況下都是由系統(tǒng)直接調(diào)用類的拷貝構(gòu)造函數(shù)而不是構(gòu)造函數(shù)。

注意:C=D;不會(huì)調(diào)用拷貝構(gòu)造函數(shù),這種情況下使用的是重載‘=’運(yùn)算符的方法。(詳見(jiàn)運(yùn)算符重載);

由于C中定義struct變量的時(shí)候,使用的全部是指針,不會(huì)用到拷貝構(gòu)造函數(shù),所以暫不考慮。對(duì)于原來(lái)函數(shù)參數(shù)或者返回值需要類變量的,要全部轉(zhuǎn)化為類指針的方式。實(shí)例化類時(shí)作參數(shù)的情況,可以通過(guò)另外定義一個(gè)帶參數(shù)的構(gòu)造函數(shù)來(lái)解決。

五.類的內(nèi)聯(lián)函數(shù)和虛函數(shù)


內(nèi)聯(lián)函數(shù)和虛函數(shù)的修飾符inline 、virture 要全部去掉。內(nèi)聯(lián)函數(shù)體則要去掉,將內(nèi)聯(lián)函數(shù)在外面定義成一個(gè)函數(shù)。如:

class B{ … virture void funb(); inline int add()const {return a+b;};private: int a; int b; …}
改為:

typedef classB B;struct classB{ … void (*funb)(struct classB *p); int (*add)(struct classB *p); int a; int b;} void classB_funb(B *p){ …} int classB_add(B *p){ return p->a+p->b;} void fun_classB(B *p){ … p->funb=classB_funb; p->add= classB_add;}

六.重載


類中重載有函數(shù)重載和運(yùn)算符重載兩種:
1)函數(shù)的重載

函數(shù)重載滿足的條件是:函數(shù)名相同,參數(shù)個(gè)數(shù)或者參數(shù)類型不同。
這樣在調(diào)用的時(shí)候,會(huì)根據(jù)你輸入的參數(shù)不同,調(diào)用不同的函數(shù)。
在C中只好分別起不同的名字,沒(méi)有別的解決辦法。

2)運(yùn)算符重載

運(yùn)算符重載只是為了滿足一般的運(yùn)算符使用的習(xí)慣而又不會(huì)出現(xiàn)錯(cuò)誤。
C中不支持運(yùn)算符重載,可以定義一個(gè)函數(shù)實(shí)現(xiàn)該功能。
這是一般類的修改。

七.類的繼承


1)單繼承

如果類之間有繼承關(guān)系,先將基類按照一般類的改法,修改好。然后將基類的定義部分全部拷到子類的前頭。除了將基類的構(gòu)造函數(shù)名改為子類構(gòu)造函數(shù)名外,不可以將基類定義的部分作其他改動(dòng)。并在構(gòu)造函數(shù)里調(diào)用基類的構(gòu)造函數(shù),然后如果子類覆蓋了基類的函數(shù),則要把該函數(shù)指針重定向到子類函數(shù)。這是為了保持類的繼承帶來(lái)的動(dòng)態(tài)聯(lián)編的特性。

類之間的繼承關(guān)系是復(fù)雜且多變的,為了保證基類在所有子類中的唯一而且方便修改,最好的方法就是把基類的結(jié)構(gòu)體部分做成宏,在子類中直接使用即可。

2)多繼承

我個(gè)人認(rèn)為多繼承是最好不要用,他會(huì)帶來(lái)一些問(wèn)題,會(huì)出現(xiàn)多個(gè)繼承路徑的問(wèn)題。除非是為了方便編程而使用的,如繼承接口等等。

多繼承也是可以改的,將多個(gè)基類的成員全部拷到子類里,遇到重復(fù)的成員名,則在前面加上前綴來(lái)區(qū)別,當(dāng)然這個(gè)指的是基類之間有相同的,如果是派生類和基類之間有重名的,則會(huì)覆蓋基類。

八.其他


以上就是C++中主要的與C的區(qū)別最大而且最常用的特性及修改方法。其他的還有一些比如模板的使用等等,這些都是為了方便編程,復(fù)用代碼。C中沒(méi)有,只好自己寫多個(gè)函數(shù)來(lái)分別實(shí)現(xiàn)。另外還有參數(shù)列表里的&符號(hào)要用指針替代,缺省值也要去掉,而在調(diào)用的時(shí)候要注意將缺省值寫上。


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • C語(yǔ)言
    +關(guān)注

    關(guān)注

    180

    文章

    7605

    瀏覽量

    137005
  • C++
    C++
    +關(guān)注

    關(guān)注

    22

    文章

    2110

    瀏覽量

    73688
  • 源代碼
    +關(guān)注

    關(guān)注

    96

    文章

    2945

    瀏覽量

    66778

原文標(biāo)題:手把手教你:如何把C++的源程序改寫成C語(yǔ)言

文章出處:【微信號(hào):wujianying_danpianji,微信公眾號(hào):?jiǎn)纹瑱C(jī)精講吳鑒鷹】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    AKI跨語(yǔ)言調(diào)用庫(kù)神助攻C/C++代碼遷移至HarmonyOS NEXT

    /C++代碼快速遷移至HarmonyOS NEXT。憑借卓越的兼容性,AKI已成為廠商與開(kāi)發(fā)者打造鴻蒙原生應(yīng)用過(guò)程中廣泛使用的跨語(yǔ)言調(diào)用解決方案。 AKI是一款專為鴻蒙原生開(kāi)發(fā)設(shè)計(jì)的FFI(外部函數(shù)接口
    發(fā)表于 01-02 17:08

    同樣是函數(shù),在CC++中有什么區(qū)別

    同樣是函數(shù),在 CC++ 中有什么區(qū)別? 第一個(gè)返回值。 C語(yǔ)言的函數(shù)可以不寫返回值類型,編譯器會(huì)默認(rèn)為返回 int。 但是 C++
    的頭像 發(fā)表于 11-29 10:25 ?334次閱讀

    C語(yǔ)言C++中結(jié)構(gòu)體的區(qū)別

    同樣是結(jié)構(gòu)體,看看在C語(yǔ)言C++中有什么區(qū)別?
    的頭像 發(fā)表于 10-30 15:11 ?247次閱讀

    C7000優(yōu)化C/C++編譯器

    電子發(fā)燒友網(wǎng)站提供《C7000優(yōu)化C/C++編譯器.pdf》資料免費(fèi)下載
    發(fā)表于 10-30 09:45 ?0次下載
    <b class='flag-5'>C</b>7000優(yōu)化<b class='flag-5'>C</b>/<b class='flag-5'>C++</b>編譯器

    技術(shù)干貨驛站 ▏深入理解C語(yǔ)言:掌握程序結(jié)構(gòu)知識(shí)

    在計(jì)算機(jī)編程的世界中,C語(yǔ)言被廣泛認(rèn)可為一門強(qiáng)大而高效的編程語(yǔ)言,其簡(jiǎn)潔的語(yǔ)法和直接的指令使得它成為了許多程序員的首選。了解C
    的頭像 發(fā)表于 07-27 08:45 ?1427次閱讀
    技術(shù)干貨驛站 ▏深入理解<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>:掌握<b class='flag-5'>程序</b>結(jié)構(gòu)知識(shí)

    C++語(yǔ)言基礎(chǔ)知識(shí)

    電子發(fā)燒友網(wǎng)站提供《C++語(yǔ)言基礎(chǔ)知識(shí).pdf》資料免費(fèi)下載
    發(fā)表于 07-19 10:58 ?7次下載

    SEGGER編譯器優(yōu)化和安全技術(shù)介紹 支持最新CC++語(yǔ)言

    SEGGER編譯器是專門為ARM和RISC-V微控制器設(shè)計(jì)的優(yōu)化C/C++編譯器。它建立在強(qiáng)大的Clang前端上,支持最新的CC++語(yǔ)言
    的頭像 發(fā)表于 06-04 15:31 ?1480次閱讀
    SEGGER編譯器優(yōu)化和安全技術(shù)介紹 支持最新<b class='flag-5'>C</b>和<b class='flag-5'>C++</b><b class='flag-5'>語(yǔ)言</b>

    keil用c++編譯含有rtos模塊時(shí)的錯(cuò)誤問(wèn)題怎么解決?

    近期看到一個(gè)哥們用c++寫嵌入式,感覺(jué)蠻有趣,如果當(dāng)初我的程序要是能用類就輕松多了,所以也想嘗試一下。雖然不會(huì)c++但是還是多少會(huì)用簡(jiǎn)單的class,這個(gè)c#上用了不少。我的工程包括u
    發(fā)表于 05-09 08:29

    C/C++中兩種宏實(shí)現(xiàn)方式

    #ifndef的方式受C/C++語(yǔ)言標(biāo)準(zhǔn)支持。它不僅可以保證同一個(gè)文件不會(huì)被包含多次,也能保證內(nèi)容完全相同的兩個(gè)文件(或者代碼片段)不會(huì)被不小心同時(shí)包含。
    的頭像 發(fā)表于 04-19 11:50 ?645次閱讀

    為什么很少用C++開(kāi)發(fā)單片機(jī)

    C語(yǔ)言是面向過(guò)程的語(yǔ)言,C++是面向?qū)ο蟮木幊?b class='flag-5'>語(yǔ)言。結(jié)合本文來(lái)說(shuō),面向過(guò)程相比面向?qū)ο蟮木幊?,生成代碼量(bin文件)更小,運(yùn)行效率更高。
    發(fā)表于 03-25 14:26 ?1014次閱讀
    為什么很少用<b class='flag-5'>C++</b>開(kāi)發(fā)單片機(jī)

    C語(yǔ)言#define的應(yīng)用

    C/C++ 編程語(yǔ)言中,當(dāng)程序被編譯時(shí),被發(fā)送到編譯器,編譯器將程序轉(zhuǎn)換為機(jī)器語(yǔ)言,然后完成
    發(fā)表于 03-06 11:29 ?388次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>#define的應(yīng)用

    谷歌捐款100萬(wàn)美元給Rust基金會(huì),以增強(qiáng)C++與Rust的交互性

    如今,谷歌多項(xiàng)核心業(yè)務(wù)仍以 C++為主要編程語(yǔ)言,雖然無(wú)法直接使用Rust替代現(xiàn)有的C++程序,但谷歌依然選擇支持Rust基金會(huì)的“Interop Initiative”計(jì)劃,幫助那些
    的頭像 發(fā)表于 02-19 15:41 ?665次閱讀

    c語(yǔ)言,c++,java,python區(qū)別

    C語(yǔ)言、C++、Java和Python是四種常見(jiàn)的編程語(yǔ)言,各有優(yōu)點(diǎn)和特點(diǎn)。 C語(yǔ)言
    的頭像 發(fā)表于 02-05 14:11 ?2431次閱讀

    vb語(yǔ)言c++語(yǔ)言的區(qū)別

    VB語(yǔ)言C++語(yǔ)言是兩種不同的編程語(yǔ)言,雖然它們都屬于高級(jí)編程語(yǔ)言,但在設(shè)計(jì)和用途上有很多區(qū)別。下面將詳細(xì)比較VB
    的頭像 發(fā)表于 02-01 10:20 ?2368次閱讀

    C++簡(jiǎn)史:C++是如何開(kāi)始的

    的 MISRA C++:2023 博客系列的第二部分。 在這篇博客中,我們將深入探討 C++ 的歷史、編程語(yǔ)言多年來(lái)的發(fā)展歷程以及它的下一步發(fā)展方向。
    的頭像 發(fā)表于 01-11 09:00 ?617次閱讀
    <b class='flag-5'>C++</b>簡(jiǎn)史:<b class='flag-5'>C++</b>是如何開(kāi)始的