C语言面向对象的艺术

编程语言及工具

105人已加入

描述

先来看封装。

所谓封装,通俗地说,就是一个姑娘化了妆,只给你看她想让你看的那一面,至于里面是否刮了骨、垫了东西,不给你看。说到封装就得说隐藏,这是对兄弟概念;其实我理解隐藏是更深的封装,完全不给你看见,而封装可能是犹抱琵琶半遮面。封装在 C++ 语言中有 protected 、 private 关键字在语言层面上支持,而 C 语言中没有这些。C 有结构体( struct ),其实可以实现封装和隐藏。

struct A_private;
struct A {
    int a;
    ...
    void (*func)(struct A*);

    struct A_private * priv;
};

上面的代码,我们只向前声明结构体 struct A_private ,没人知道它里面具体是什么东西。假如 struct A 对应的头文件是 A.h ,那么把 A_private 的声明放在 A_p.h 中,A.c 文件包含 A_p.h ,那么在实现 struct A 的函数指针 func 时如何使用 struct A_private ,客户程序员根本无须知道。

这样做的好处是显而易见的,除了预定义好的接口,客户程序员完全不需要知道实现细节,即便实现经过重构完全重来,客户程序员也不需要关注,甚至相应的模块连重新编译都不要——因为 A.h 自始至终都没变过。

另外对于某些方法,仅仅是在对象内部使用,它们将采用static修辞把作用范围局限在一个文件的内部:

到现在为止,封装和隐藏就实现了,而且很彻底。

再看继承。

首先C语言中的void指针是非常强大的,不仅可以指向变量而且还可以泛指向任何数据,例如不同类型的变量,函数等。不同变量间的指针只需要强制转换类型就可实现不同类型数据的访问。例如如下代码:

struct parent_class
{
int a, b;
char *str;
};
struct child_class
{
struct parent_class p;
int a, b;
};
void func()
{
struct child_class obj, *obj_ptr;
struct parent_class *parent_ptr;
obj ptr = &obj;
/* 获得父指针 */
parent ptr = (struct parent*) &obj;
parent ptr- >a = 1;
parent ptr- >b = 5;
obj ptr- >a = 10;
obj ptr- >b = 100;
}

在上面代码中,注意subobject结构中第一个成员p,这种声明方式代表subobject类型的数据中开始的位置包含一个parent类型的变量。在函数func中obj是一个subobject对象,正向这个结构类型指示的,它前面的数据应该包含一个parent类型的数据。在第17行的强制类型赋值中parent ptr指向了obj变量的首地址,实际上也就是obj变量中的p对象。

好了,现在parent ptr指向的是一个真真实实的parent类型的结构,那么可以按照parent的方式访问其中的成员,当然也包括可以使用和parent结构相关的函数来处理内部数据,因为一个正常的,正确的代码,它是不会越界访问parent结构体以外的数据的。经过这基本的结构体层层相套包含,对象简单的继存关系就体现出来了:父对象放于数据块的最前方,代码中可以通过强制类型转换来获得父对象指针。

最后来看多态。

这里只讲解多态最基本的用法,就是实现改写对象的行为。虚函数与重载涉及比较复杂的底层机制,笔者暂时还没有研究清楚。

多态,就是说用同一的接口代码处理不同的数据。比如说,下面的base_class结构就是一个通用的数据结构,我们也不清楚a是什么数据,vfunc是什么处理函数?但是,我们处理的时候只要调用base_class->vfunc(int a)就可以了。剩下来的事情我们不需要管,因为不同的接口会有不同的函数去处理,我们只要学会调用就可以了。

struct base_class
{
int a;
void (*vfunc)(int a);
}
void base_class_vfunc(struct base_class *self, int a)
{
assert(self != NULL);
assert(slef- >vfunc != NULL);
self- >vfunc(a);
}
struct child_class
{
struct base_class parent;
int b;
};
void child_class_init(struct child_class* self)
{
struct base_class* parent;
parent = (struct_base_class*) self;
assert(parent != NULL);
parent- >vfunc = child_class_vfunc;
}
static void child_class_vfunc(struct child_class*self, int a)
{
self- >b = a + 10;
}
打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分