電子工程網
標題: 如何把C++的源程序改寫成C語言 [打印本頁]
作者: XYD李 時間: 2021-8-5 16:23
標題: 如何把C++的源程序改寫成C語言
由于C++解釋器比C語言解釋器占用的存儲空間要大500k左右。為了節省有限的存儲空間,降低成本,同時也為了提高效率,將用C++語言寫的源程序用C語言改寫是很有必要的。
C++與C區別最大的就是C++中的類的概念和特性,將C++改為C的問題,就轉換成如何將類化去的問題。
方法有兩種:
第一種是將C++中的面向對象特征去掉,先全部理解源代碼的邏輯,然后改寫;第二種是在C中保留面向對象的部分特征,用結構體實現類的功能。
第一種方法,對于類的數目很少的情況還可以,如果類的數目比較多,全部理解源代碼,然后重寫就很耗時間,而且很容易出錯,更甚者,如果遇到大的項目想全部理解源代碼幾乎是不可能的。
hpijs程序中類有140多個,這個時候就需要采用第二個方法了,你可以一個類一個類的改沒有什么太高的難度,如果不是筆誤的話,幾乎不會出錯,而且根本不需要理解程序邏輯,也許改完后你對程序所要實現的功能還一無所知。倒不是說一無所知對大家有好處,只是想說這種方法的與程序邏輯本身的無關性。
下面對C++的一些特性,以及如何在c里實現或者替代,作一些初步的探討:
說明:
函數Ixx為類xx的構造函數的實現。
原類的成員函數改為前綴為結構體名+‘_’的函數。
函數指針U為原類的析構函數的聲明;
U+結構體名稱為原類的析構函數的實現;
Fun-_+結構體名為對該結構體成員函數指針進行指向;
以后遇到上述情況將不再說明。
一.類的成員函數和數據成員
由于struct沒有對成員的訪問權限進行控制,必須加入額外的機制進行訪問控制,這樣一來就使得程序復雜化了,所以只能放棄訪問權限的控制。
1)對于類的數據成員可以直接轉為C中結構體的數據成員。
2)函數則需轉化為對應的函數指針,因為struct里不允許出現函數的聲明和定義。而函數前如果有virture,inline等修飾符也要去掉,如函數void funca(int a);改為void (*funca)(struct B *p,int a);大家可以看到函數指針的原型里加了一個指針struct B的指針,這是因為要在函數內部對類的成員進行操作,要靠該指針指定結構體的成員。在類的成員函數里,實際上在參數列里也隱含有一個指向自身的this指針。
3)對于靜態成員則要定義成全局變量或全局函數,因為結構體中不能有靜態成員。
二.類的構造函數
類在實例化的時候會調用類的缺省構造函數,在struct里,要定義一個同名函數指針指向一個具有構造函數功能的初始化函數,與構造函數不同的是,要在初始化函數里加入進行函數指針初始化的語句.使用的時候在創建結構體變量的時候要用malloc而不是new,并且這個時候要手工調用初始化函數。
如下例所示:
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);//構造函數指針
void (*U)(struct classA *p);//析構函數指針
void (*func)(struct classA *p,int a);
int b;
};
void fun_A(A *p)
{
p->func=classA_func; //將函數指針初始化
}
void IA(A *p) //構造函數,命名規則在類名前加I
{
fun_A(p);
p->b=0; //原構造函數所作部分
}
void classA_func(A *p,int a)
{
p->b=a;
}
在使用的地方采用如下方式:
A *s=(A*)malloc(sizeof(A));
s->A=IA;
s->A(s);
三.類的析構函數
類的析構函數所作的工作是釋放所占的資源。
在C中,無論是哪個struct都用函數指針U替代析構函數。之所以所有的struct都用指針U是基于如下情況:
如果將子類指針賦給基類指針,基類指針在釋放的時候不必考慮調用哪個函數名的析構函數,只需調用成員函數U即可。成員函數U需要像一般成員函數一樣在fun_類名()函數中指定。
類的析構函數是由系統調用的,在C中則要顯式調用。至于何時調用,要準確判斷。
四.類的拷貝構造函數
類的拷貝構造函數主要用途是加快以下情況下類的構建速度:
1. 作為參數傳給函數。(additem(Itema))
2. 作為函數返回值。
3. 實例化類時作參數。
這三種情況下都是由系統直接調用類的拷貝構造函數而不是構造函數。
注意:C=D;不會調用拷貝構造函數,這種情況下使用的是重載‘=’運算符的方法。(詳見運算符重載);
由于C中定義struct變量的時候,使用的全部是指針,不會用到拷貝構造函數,所以暫不考慮。對于原來函數參數或者返回值需要類變量的,要全部轉化為類指針的方式。實例化類時作參數的情況,可以通過另外定義一個帶參數的構造函數來解決。
五.類的內聯函數和虛函數
內聯函數和虛函數的修飾符inline 、virture 要全部去掉。內聯函數體則要去掉,將內聯函數在外面定義成一個函數。如:
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;
}
六.重載
類中重載有函數重載和運算符重載兩種:
1)函數的重載
函數重載滿足的條件是:函數名相同,參數個數或者參數類型不同。
這樣在調用的時候,會根據你輸入的參數不同,調用不同的函數。
在C中只好分別起不同的名字,沒有別的解決辦法。
2)運算符重載
運算符重載只是為了滿足一般的運算符使用的習慣而又不會出現錯誤。
C中不支持運算符重載,可以定義一個函數實現該功能。
這是一般類的修改。
課程咨詢:C語言,單片機,Linux電路設計,PCB軟件測試,python,JAVA,C++,QT等課程培訓提升,面授線上學習,有需要加18025267692(微信)
七.其他
以上就是C++中主要的與C的區別最大而且最常用的特性及修改方法。其他的還有一些比如模板的使用等等,這些都是為了方便編程,復用代碼。C中沒有,只好自己寫多個函數來分別實現。另外還有參數列表里的&符號要用指針替代,缺省值也要去掉,而在調用的時候要注意將缺省值寫上。
還想看到哪些干貨,可以留言告訴小編哦!
歡迎光臨 電子工程網 (http://m.qingdxww.cn/) |
Powered by Discuz! X3.4 |
主站蜘蛛池模板:
在线播放毛片
|
青草午夜精品视频在线观看
|
日韩欧美在线一级一中文字暮
|
国产精品手机视频一区二区
|
成人精品视频一区二区三区尤物
|
午夜色网站
|
国产一区三区二区中文在线
|
粉嫩极品国产在线观看
|
韩国福利视频一区二区
|
国产一区二区三区免费观看
|
亚洲视频自拍偷拍
|
九九99国产精品视频
|
日本经典在线三级视频
|
色综合合久久天天给综看
|
视频一区二区在线播放
|
天天摸夜夜摸成人免费视频
|
97精品一区二区三区在线不卡
|
美女日日日
|
日韩视
|
国产一区二区免费在线
|
国产精品四虎
|
偷窥自拍p|
激情五月开心婷婷
|
无码h肉动漫在线观看
|
香蕉尹人
|
欧美亚洲综合另类在线观看
|
影音先锋色先锋女同另类
|
欧美高清360|
成人精品一区二区三区中文字幕
|
妖精视频一区二区三区
|
日本免费一级
|
欧美3p在线观看一区二区三区
|
国产一区亚洲二区三区
|
videos欧美hd精品
|
日韩精品视频在线观看免费
|
97视频免费
|
99er视频|
久久久国产精品免费
|
国产黄网站在线观看
|
国产区免费|
亚洲短视频在线观看
|