如摘要所说,C语言不支持OOP(面向对象的编程)。并这不意味着我们就不能对C进行面向对象的开发,只是过程要复杂许多。原来以C++的许多工作,在C语言中需我们手动去完成。

博主将与大家一起研究一下如下用C语言实现面象对象的编程。

面向对象的三大特性:封装、继承、多态

我们要达到的目的如下:

Animal是动物,有两个方法:Eat()吃,Breed()繁衍。

Bird与Mammal都是Animal,Mammal是哺乳动物。

Penguin是企鹅,企鹅是Bird,企鹅不会飞。

Swallow是燕子,是Bird,会飞。

Bat是蝙蝠,是Mammal,会飞。

Tiger是老虎,是Mammal,不会飞。

Plane是飞机,会飞。它不是动物。

从上面的类继承关系来看,由于Swallow, Bat, Plane会飞,所以它们都继承了IFly接口。

首先我们用C++的类来实现上面的关系:

class Animal {
public:
virtual void Eat() = 0;
virtual void Breed() = 0;
}; class Bird : public Animal {
public:
virtual void Breed() {
cout << "蛋生" << endl;
}
}; class Mammal : public Animal {
public:
virtual void Breed() {
cout << "胎生" << endl;
}
}; class IFly {
public:
virtual void Fly() = 0;
}; class Penguin : public Bird {
public:
virtual void Eat() {
cout << "企鹅吃鱼" << endl;
}
}; class Swallow : public Bird , public IFly {
public:
virtual void Eat() {
cout << "燕子吃虫子" << endl;
}
virtual void Fly() {
cout << "燕子飞呀飞" << endl;
}
}; class Bat : public Mammal, public IFly {
public:
virtual void Eat() {
cout << "蝙蝠吃飞虫" << endl;
}
virtual void Fly() {
cout << "蝙蝠飞呀飞" << endl;
}
}; class Tiger : public Mammal {
virtual void Eat() {
cout << "老虎吃肉" << endl;
}
}; class Plane : public IFly {
public:
virtual void Fly() {
cout << "飞机飞过天空" << endl;
}
};

用下面的main.cpp来测试它们的继承效果:

int main()
{
Penguin *penguin = new Penguin;
Swallow *swallow = new Swallow;
Bat *bat = new Bat;
Tiger *tiger = new Tiger;
Plane *plane = new Plane; Animal* animals[4] = {penguin, swallow, bat, tiger};
IFly* flies[3] = {swallow, bat, plane}; for (int i = 0; i < 4; ++i) {
animals[i]->Eat();
animals[i]->Breed();
} cout << "-------------" << endl;
for (int i = 0; i < 3; ++i)
flies[i]->Fly(); delete penguin;
delete swallow;
delete bat;
delete tiger;
delete plane; return 0;
}

执行的效果是:

企鹅吃鱼
蛋生
燕子吃虫子
蛋生
蝙蝠吃飞虫
胎生
老虎吃肉
胎生
-------------
燕子飞呀飞
蝙蝠飞呀飞
飞机飞过天空

上面演示的就是C++的多态功能。


多态这个特性给我们软件灵活性带来了很大的便利。由于某此限制,如硬件资源不够充裕、开发环境不支持C++等原理,我们不能使用C++。

那么我们下面要讨论的是用C来重新实现上面的多态功能。

main.c大致是这样子的:

int main()
{
Object* penguin = Penguin_New();
Object* swallow = Swallow_New();
Object* bat = Bat_New();
Object* tiger = Tiger_New();
Object* plane = Plane_New(); Object* animal[4] = {penguin, swallow, bat, tiger}; IFly* flies[3] = {NULL};
flies[0] = Swallow_AsIFly(swallow);
flies[1] = Bat_AsIFly(bat);
flies[2] = Plane_AsIFly(plane); for (int i = 0; i < 4; ++i) {
Animal_Eat(animal[i]);
Animal_Breed(animal[i]);
} for (int i = 0; i < 4; ++i) {
IFly_Fly(flies[i]);
} Penguin_Delete(penguin);
Swallow_Delete(swallow);
Bat_Delete(bat);
Tiger_Delete(tiger);
Plane_Delete(plane); return 0;
}

上面编译时需要加 "-std=c99" 才能通过编译。


博主已实现了上面的Demo,代码已提交到:http://git.oschina.net/hevake_lcj/C_OOP_DEMO

该Demo实现了OOP的类继承、多态的特性。继承只支持单继承,还没有实现接口功能。

每个对象由三部分组成:info, data, func

  • info,类信息,存储该对象的:类型ID、虚函数表地址

  • data,对象的数据

  • func,虚函数指针

如下为 info 的定义:

typedef struct {
uint32_t tag; //! 高16位为TAG,低16位为class_id
void* vfun; //! 虚函数结构体地址
} class_info_t;

例如 Animal 类的定义,见 animal_def.h :

typedef struct {
int health;
} Animal_Data; typedef struct {
void (*Eat)();
void (*Breed)();
} Animal_Func; typedef struct {
class_info_t info;
Animal_Data data;
Animal_Func func;
} Animal;

结构图:

<明天再写>

即将讨论话题:

- 如何表述类的继承关系?

- 为什么要将data与func分开?

博主自己测试了一下,结果是:

$ ./c_oop_demo
start
企鹅吃鱼
蛋生
燕子吃虫子
蛋生
蝙蝠吃飞虫
胎生
老虎要吃肉
胎生
end

从上看来,已达到了预期的多态效果。

C++与C的比较

居说C++编译出来的可执行文件远多于C。于是博主对比了一下c_oop_demo与C++编译的同功能的可执行文件cpp_demo。博主惊讶地发现 c_oop_demo 的文件大小既还比 cpp_demo大一点。况且上面的 c_oop_demo 还没有实现接口功能呢,要是实现了,那不更大?这不由令博主对用C实现OOP,以为可以节省空间的想法大为失望。

看来,在实现同样的oop功能下,C++编译出的输出文件比自己手把手写的c_oop_demo要小,说明C++在这方便做了不少的优化。相比之下,C++用50多行的代码实现的功能,用C(博主亲自统计的)居然要写近1000行代码。代码的可维护性远不及C++。说C++生成的文件庞大,真是冤枉了C++。用C完成同等功还不如C++干得漂亮。

https://yq.aliyun.com/articles/33326?spm=5176.100239.blogrightarea33100.17.9NBJrk

用C实现OOP面向对象编程(1)的更多相关文章

  1. Java实现OOP(面向对象编程)

    一.对象的综述 面向对象编程(OOP)具有多方面的吸引力.对管理人员,它实现了更快和更廉价的开发与维护过程.对分析与设计人员,建模处理变得更加简单,能生成清晰.易于维护的设计方案.对程序员,对象模型显 ...

  2. 一百零六、SAP的OOP面向对象编程,OO-ALV的简介

    面向对象编程,如图 基本概念: 1.对象(Object)是一个现实实体的抽象.一个对象可被认为是一个把数据(属性)和程序(方法)封装在一起的实体,这个程序产生该对象的动作或对它接受到的外界信号的反应. ...

  3. Swift -POP( 面向协议编程)与OOP(面向对象编程)

    面向协议编程(Protocol Oriented Programming,简称POP),是Swift的一种编程范式,Apple于2015年WWDC提出的,如果大家看Swift的标准库,就会看到大量PO ...

  4. (转)OOP(面向对象编程)的几大原则

    文章转载自:http://blog.csdn.net/anders_zhuo/article/details/8949566 设计模式遵循的一般原则: 1.开-闭原则(Open-Closed Prin ...

  5. OOP(面向对象编程)的一些特性

    接口:接口是把公共实例(非静态)方法和属性结合起来,以封装特定功能的一个集合.一旦定义了接口,就可以在类中实现它.接口注意事项:接口不能单独存在.不能像实例化一个类那样实例化接口.另外,接口不能包含实 ...

  6. OOP面向对象编程(下)

    我们怎么去模拟重载,在javasceipr中我们可以通过参数的类型区别或者数量的区别,来去让同样一个函数名字,可以根据不同的参数列表的情况来去调用相应的函数. javascript中函数类型是不确定的 ...

  7. javascript OOP 面向对象编程

    Pseudo-class declaration 原文地址:http://javascript.info/tutorial/pseudo-classical-pattern#pseudo-class- ...

  8. 重学C++ (十一) OOP面向对象编程(2)

    转换与继承 本节主要须要区分的是: 基类和派生类的转换: 引用(指针)的转换和对象的转换. 1.每一个派生类对象包括一个基类部分.因此.能够像使用基类对象一样在派生类对象上执行操作. 基于这一点,能够 ...

  9. 洗礼灵魂,修炼python(31)--面向对象编程(1)—面向对象,对象,类的了解

    面向对象 1.什么是面向对象 (图片来自网络) 哈哈,当然不是图中的意思. 1).面向对象(Object Oriented,OO)是软件开发方法.利用各大搜索引擎得到的解释都太官方,完全看不懂啥意思对 ...

随机推荐

  1. svg查看预览 , 鼠标控制放大缩小 , 托拉拽等

    自己写是不可能了 , 所以要借用插件 svg-panzoom.js 地址 : https://github.com/ariutta/svg-pan-zoom#demos 及常见问题https://ww ...

  2. Data analysis system

    A data analysis system, particularly, a system capable of efficiently analyzing big data is provided ...

  3. JavaScript实现算法

    leetcode算法题(JavaScript实现)   题外话 刷了一段时间的codewars的JavaScript题目之后,它给我最大的感受就是,会帮助你迅速的提升你希望练习的语言的API的熟悉程度 ...

  4. vs2008C1902数据库管理程序不匹配

    打开一大早vs2008,有这么奇怪的错误, 删了dll正好.图. 版权声明:本文博主原创文章.博客,未经同意不得转载.

  5. Win7 32bit下一个hadoop2.5.1源代码编译平台的搭建各种错误遇到

    从小白在安装hadoop困难和错误时遇到说起,同时,我们也希望能得到上帝的指示. 首先hadoop更新速度非常快,最新的是hadoop2.5.1,因此就介绍下在安装2.5.1时遇到的各种困难. 假设直 ...

  6. Qt 的几个核心机制总结之 布局(QWidget可以设置setSizePolicy,而QSizePolicy有Fixed,minimum,maximum,preferred,expanding,ignore等7个属性,还可以横竖分开)

    1.Qt布局的作用 Qt的布局是通过布局管理器来实现的,布局管理器负责在父类窗口部件区域构建子窗口部件,使得放置在窗体中的每个窗口部件都有一个适合的大小和位置,并且能够随着应用程序本身的变化而变化从而 ...

  7. 深入理解最强桌面地图控件GMAP.NET --- 街景地图(StreetView)

    原文:深入理解最强桌面地图控件GMAP.NET --- 街景地图(StreetView) 很久没有更新博客了,今天无事把GMAP.NET的代码又重新翻了翻,看到了街景地图的例子. 街景地图是谷歌最早提 ...

  8. 【C语言学习】C语言功能

    代码,功能为了更好地实现模块化编程.那么,什么是函数的性质?在函数中定义的变量(全局变量.局部变量.静态变量)如何存储?为什么范围和全局变量和局部变量的寿命是不一样的?只是有一个更深入的了解的功能.能 ...

  9. CORSFilter

    import java.io.IOException; import javax.servlet.Filter;import javax.servlet.FilterChain;import java ...

  10. Cocos2d-x 3.0final 终结者系列教程09-漆节点Node中间Schedule

    怎么做HelloWorld工程HelloWorld文字实现它自己主动运动? 有的童鞋会想到使用线程.不断的变化Label的Position, 不要那样做,因为Cocos2d-x在主线程只能被改变Nod ...