我要努力工作,加油!

C语言陷阱与技巧27节,“函数指针结构体”为C语言找了个“对象”

		发表于: 2019-06-26 07:52:54 | 已被阅读: 50 | 分类于: C语言
		

上一节讨论了C语言中的指针可以看作是一种普通的数据类型,这么一来,函数指针数组就不难理解了,无非就是存放函数指针元素的数组而已。

函数指针结构体

稍微思考一下,应该能够想到C语言中的普通数据类型不仅可以用于定义数组,还可以用来定义结构体,例如:

struct cn{
    char    c;
    int     i;
    double  d;
};

那么可以看作“普通数据类型”的函数指针也可以定义结构体吗?自然是可以的,请看下面的C语言代码示例:

struct cfun{
    void    (*vfun)();
    int     (*ifun)(int a);
    float   (*ffun)(float x);
};

显然,使用“函数指针类型”定义结构体和使用普通数据类型没什么两样。

C语言的“对象”

事实上,定义好函数指针结构体之后,用起来也和普通数据类型定义的结构体一样:

struct cfun s;
// init s
s.vfun();
s.ifun(a);
struct cfun *ps = &s;
ps->ffun(x);

从上面这几行C语言代码可以看出,使用函数指针结构体调用函数,看起来很像其他高级语言中的“面向对象”风格。的确如此,C语言中的结构体和指针语法,允许C语言程序员写出“面向对象”风格的代码。

很多程序员觉得C语言没有对象语法,认为C语言只能按照面向过程风格的代码开发程序。其实,“面向对象”更多时候是一种编程思想,而不仅限于一种编程语言的语法。

不过,因为C语言没有原生的对象语法,在使用函数指针结构体之前,必须对结构体初始化,否则最终C语言程序就会有崩溃的风险。因为和其他变量一样,没有初始化的函数指针变量的指向是不确定的,没有初始化过的函数指针本质上是一种“野指针”,操作野指针的危害相信读者已经了解。

初始化C语言的“对象”

使用函数指针定义的结构体成员本身不具备功能,它只是一个指针,只有将其指向某个具体函数,它才会具备实际功能。下面是一个例子:

void myprint()
{
    printf("hello embedTime\n");
}
int add_inum(int a)
{
    return a+2;
}
float add_fnum(float x)
{
    return x+2.0;
}
void init_cfun(struct cfun *s)
{
    s->vfun = myprint;
    s->ifun = add_inum;
    s->ffun = add_fnum;
}

struct cfun s;
init_cfun(&s);
// s now is available
s->vfun();  // hello embedTime

请看上述C语言代码,init_cfun() 函数将接收到的函数指针结构体初始化,即将该结构体的各个成员指向事先实现的函数。只有在 init_cfun(&s); 之后,结构体 s 才可用。

当然,我们也可以基于C语言结构体的赋值语法,实现一套静态初始化的代码,请看:

static struct cfun CFUN_INIT = {myprint, add_inum, add_fnum};

之后再定义 cfun 结构体时,直接将 CFUN_INIT 赋值给该结构体,就可以实现初始化了,请看相关C语言代码:

struct cfun s = CFUN_INIT;
// s now is available
s->vfun();  // hello embedTime
很多C语言程序员在处理结构体赋值时,常使用 memcpy() 拷贝内存,其实对于相同的结构体,直接赋值也是一样的。

C语言“对象”的意义

基于C语言指针和结构体语法实现的“面向对象”风格代码,有利于程序员管理。在实际C语言项目开发中,某个模块的功能常常无法只依靠一个函数实现,若干个函数实现一个模块的功能并不少见。但是这些函数零散的分布在整个C语言项目代码中,管理起来有时的确比较头疼。

读完本节,相信读者应该发现,结合C语言指针和结构体语法,能够方便的将一个模块的所有函数接口封装成一个“对象”,一个对象管理一个模块,整个代码就显得比较简洁易懂了。事实上,在 Linux 内核代码中,基于C语言指针和结构体语法实现的“面向对象”风格代码相当常见、

另外值得说明的是,C语言的“对象”也便于函数命名。现在想象一下,加入有若干个模块都需提供读写的功能,那在实现C语言代码时,可能在项目中,会出现下面这种命名:

void moduleA_write(void *data)
{
    ...
}
void moduleB_write(void *data)
{
    ...
}
void moduleC_write(void *data)
{
    ...
}

写在C语言代码中,会重复写很多次“module”。如果使用“对象”封装,方法名的差异完全可以使用“对象名”区分,例如:

ma.write();
mb.write();

显然这种写法简洁许多。

小结

本节主要讨论了C语言中的指针与结构体语法的结合使用,可以看出,基于此C语言甚至可以模拟其他高级编程语言中的“面向对象”风格。文章最后尝试讨论了“面向对象”这种编程思想在C语言项目开发中的便利性,当然了,这里讨论的只是很小的一个点,读者在之后的实际项目开发中,一定能够发现更多特性的。