RM新时代网站-首页

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

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

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

如何把C++程序改寫成C語言

STM32嵌入式開發(fā) ? 來源:STM32嵌入式開發(fā) ? 2023-04-21 10:12 ? 次閱讀

由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間、降低成本,同時也為了提高效率,將用C++語言寫的源程序用C語言改寫是很有必要的。

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

方法有兩種:

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

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

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

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

說明:

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

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

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

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

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

以后遇到上述情況將不再說明

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

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

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

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

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

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

類在實例化的時候會調(diào)用類的缺省構(gòu)造函數(shù),在struct里,要定義一個同名函數(shù)指針指向一個具有構(gòu)造函數(shù)功能的初始化函數(shù),與構(gòu)造函數(shù)不同的是,要在初始化函數(shù)里加入進(jìn)行函數(shù)指針初始化的語句。使用的時候,在創(chuàng)建結(jié)構(gòu)體變量的時候要用malloc而不是new,并且這個時候要手工調(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中,無論是哪個struct都用函數(shù)指針U替代析構(gòu)函數(shù)。之所以所有的struct都用指針U是基于如下情況:

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

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

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

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

1. 作為參數(shù)傳給函數(shù)。(additem(Itema))

2. 作為函數(shù)返回值。

3. 實例化類時作參數(shù)。

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

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

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

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

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

wKgZomRB8VaAd-XpAAArEI4MxKs126.png

改為:

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ù)個數(shù)或者參數(shù)類型不同。

這樣在調(diào)用的時候,會根據(jù)你輸入的參數(shù)不同,調(diào)用不同的函數(shù)。

在C中只好分別起不同的名字,沒有別的解決辦法。

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

運(yùn)算符重載只是為了滿足一般的運(yùn)算符使用的習(xí)慣而又不會出現(xiàn)錯誤。

C中不支持運(yùn)算符重載,可以定義一個函數(shù)實現(xiàn)該功能。

這是一般類的修改。

七、類的繼承

1)單繼承

如果類之間有繼承關(guān)系,先將基類按照一般類的改法,修改好,然后將基類的定義部分全部拷到子類的前頭。 除了將基類的構(gòu)造函數(shù)名改為子類構(gòu)造函數(shù)名外,不可以將基類定義的部分作其他改動。并在構(gòu)造函數(shù)里調(diào)用基類的構(gòu)造函數(shù),然后如果子類覆蓋了基類的函數(shù),則要把該函數(shù)指針重定向到子類函數(shù)。這是為了保持類的繼承帶來的動態(tài)聯(lián)編的特性。 類之間的繼承關(guān)系是復(fù)雜且多變的,為了保證基類在所有子類中的唯一而且方便修改,最好的方法就是把基類的結(jié)構(gòu)體部分做成宏,在子類中直接使用即可。

2)多繼承

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

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

其他

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

審核編輯:湯梓紅

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

    關(guān)注

    180

    文章

    7604

    瀏覽量

    136683
  • 程序
    +關(guān)注

    關(guān)注

    117

    文章

    3785

    瀏覽量

    81002
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4327

    瀏覽量

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

    關(guān)注

    22

    文章

    2108

    瀏覽量

    73618
  • 解釋器
    +關(guān)注

    關(guān)注

    0

    文章

    103

    瀏覽量

    6509

原文標(biāo)題:把C++程序改寫成了C語言的?

文章出處:【微信號:c-stm32,微信公眾號:STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    如何C++的源程序改寫成C語言

    由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源
    發(fā)表于 07-08 20:51

    如何C++的源程序改寫成C語言

    由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源
    發(fā)表于 07-11 09:34

    如何C++的源程序改寫成C語言

    由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源
    發(fā)表于 06-30 10:54

    如何C++的源程序改寫成C語言

    由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源
    發(fā)表于 07-05 14:59

    如何C++的源程序改寫成C語言

    `由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源
    發(fā)表于 07-07 09:31

    如何C++的源程序改寫成C語言

    由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節(jié)省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源
    發(fā)表于 11-17 10:16

    c++程序設(shè)計教程

    C++是一種高效應(yīng)用的程序設(shè)計語言,它即可進(jìn)行過程化程序設(shè)計,也可進(jìn)行面向?qū)ο?b class='flag-5'>程序設(shè)計,因而成了編程人員最廣泛使用的工具。學(xué)好C++,很容易
    發(fā)表于 04-08 00:36 ?124次下載
    <b class='flag-5'>c++</b><b class='flag-5'>程序</b>設(shè)計教程

    c++程序設(shè)計語言(特別版)

    c++程序設(shè)計語言介紹了標(biāo)準(zhǔn)C++以及由C++所支持的關(guān)鍵性編程技術(shù)和設(shè)計技術(shù)。標(biāo)準(zhǔn)C++較以前的版本功能更
    發(fā)表于 09-05 15:28 ?0次下載

    c++程序設(shè)計語言題解

    c++程序設(shè)計語言題解是與Bjarne Stroustrup的《C++程序設(shè)計語言》一書配套使用的習(xí)題解答,為從《C++
    發(fā)表于 09-05 16:00 ?121次下載
    <b class='flag-5'>c++</b><b class='flag-5'>程序設(shè)計語言</b>題解

    c++程序設(shè)計習(xí)題答案

    本書是與Bjarne Stroustrup的《C++程序設(shè)計語言》一書配套使用的習(xí)題解答,為從《C++程序設(shè)計語言》中精選出
    發(fā)表于 09-05 16:21 ?2次下載
    <b class='flag-5'>c++</b><b class='flag-5'>程序</b>設(shè)計習(xí)題答案

    C++程序設(shè)計語言-揣錦華

    C++程序設(shè)計語言課程
    發(fā)表于 04-09 18:01 ?0次下載
    <b class='flag-5'>C++</b><b class='flag-5'>程序設(shè)計語言</b>-揣錦華

    程序設(shè)計及C++語言課程設(shè)計教學(xué)大鋼

    程序設(shè)計及C++語言課程設(shè)計教學(xué)大鋼:程序設(shè)計及C++語言》課程設(shè)計教學(xué)大鋼(課程設(shè)計類) 課
    發(fā)表于 06-25 23:29 ?15次下載

    C程序設(shè)計語言第8章OO與C++、Java、C#

    C程序設(shè)計語言第8章OO與C++、Java、C# C++中的Hello World!􀂄 兼容
    發(fā)表于 06-21 09:05 ?45次下載

    如何C++的源程序改寫成C語言

    第一種是將C++中的面向?qū)ο筇卣魅サ?,先全部理解源代碼的邏輯,然后改寫;第二種是在C中保留面向?qū)ο蟮牟糠痔卣?,用結(jié)構(gòu)體實現(xiàn)類的功能。
    的頭像 發(fā)表于 05-14 10:08 ?2915次閱讀
    如何<b class='flag-5'>把</b><b class='flag-5'>C++</b>的源<b class='flag-5'>程序</b><b class='flag-5'>改寫成</b><b class='flag-5'>C</b><b class='flag-5'>語言</b>

    嵌入式程序開發(fā),C語言C++究竟應(yīng)該用哪個?

    在嵌入式軟件程序開發(fā)中,C語言無疑是最常被使用的程序語言。不過應(yīng)該明白的是,有些嵌入式硬件同時提供C語言
    發(fā)表于 11-03 14:21 ?60次下載
    嵌入式<b class='flag-5'>程序</b>開發(fā),<b class='flag-5'>C</b><b class='flag-5'>語言</b>和<b class='flag-5'>C++</b>究竟應(yīng)該用哪個?
    RM新时代网站-首页