C 語言實作獨立的類別
在前述的範例當中,我們直接將資料與函數封裝在結構當中,以形成物件,這種實作方式並沒有為類別定義獨立的結構,於是每個物件當中都會有一份所有成員函數的指標,當物件的數量很多時,這可能會浪費不少記憶體。
前述的這種實作方式,比較像是一種變形後的物件導向實作法,這種方法稱為原型 (Prototype) 導向的實作法,像是 JavaScript 就採用了類似的實作方式。如果我們要實作出像 Java 或 C# 一樣的物件導向作法,應該將類別的結構獨立出來,這樣會比較能夠規模化,而且通常可以節省記憶體。
在以下的程式中,我們將再度用 C 語言實作出這種方式,將物件與類別獨立成兩個不同結構。如此,不管我們建立幾份物件,類別物件永遠都只會有一個,請看下列程式碼。
程式實作:將類別獨立出來
檔案:polyClass.c
#include <stdio.h>
#define ShapeClassMembers(OBJ) float (*area)(struct OBJ*)
#define ShapeMembers(OBJ)
struct _Shape;
typedef struct _ShapeClass { // Shape 物件,沒有欄位
ShapeClassMembers(_Shape);
} ShapeClass;
typedef struct _Shape {
ShapeClass *class;
ShapeMembers(_Shape);
} Shape;
float ShapeArea(Shape *obj) { return 0; }
ShapeClass shapeClass = { .area = ShapeArea };
struct _Circle;
typedef struct _CircleClass {
ShapeClassMembers(_Circle);
float r;
} CircleClass;
typedef struct _Circle {
CircleClass *class;
ShapeMembers(_Circle);
float r;
} Circle;
float CircleArea(Circle *obj) { return 3.14 * obj->r * obj->r; }
CircleClass circleClass = { .area = CircleArea };
int main() {
Shape s = { .class = &shapeClass };
Circle c = { .class = &circleClass };
c.r = 3.0;
Shape *list[] = { &s, (Shape*) &c };
int i;
for (i=0; i<2; i++) {
Shape *o = list[i];
printf("s.area()=%G\n", o->class->area(o));
}
}
執行結果
D:\cp>gcc polyClass.c -o polyClass
D:\cp>polyClass
s.area()=0
s.area()=28.26