Tangwx

Tangwx

博客网站

C语言实现封装继承多态

一文搞懂怎么用 C 实现封装、继承、多态#

1、介绍#

封装、继承、多态是面向对象的重要特性,有了它们再结合一些设计模式可以让我们的代码变得高内聚、低耦合,增加代码的安全性、可读性、可维护性。那么对于 C 语言这种面向过程的编程语言,能不能也实现封装、继承、多态,从而提升代码的质量呢?答案是显而易见的,我们在linux内核代码的学习过程中,用心分析,你会发现里面已经广泛利用 C 语言实现封装、继承、多态。本篇文章会带着大家分析一下,怎么用 C 实现这些面向对象的特性,为大家后面看代码、写出高质量的代码做铺垫。

2、封装#

封装就是把一个抽象的事物的属性和相应的操作方法打包到一个类中,通过内部的方法来改变内部状态。封装的本质是隐藏信息的过程,使对象的内部状态不被外界直接访问和修改。

封装具有如下优点:

  • 提高了代码的安全性,数据只能被规定的方式访问,避免了误操作和非法访问。
  • 提高了代码的复用性,相同或类似的数据类型可以被多次利用。
  • 提高了代码的可维护性,当数据类型发生变化时,只需修改一个地方即可。

C 语言中没有类的概念,但是可以使用结构体实现对事物的封装。封装的重要意义是,将函数(方法)和数据(属性)整合在一起,数据(属性)和数据(属性)整合在一起。这样,我们就可以通过简单的一个结构指针访问到所有的数据,操作所有的函数。

以下便是一个封装的具体例子:

#include <stdio.h> 

typedef struct human
{ 
    int age; 
    char sex; 
    void (*set_age)(struct human *p, int age);
    int (*get_age)(struct human *p);
    void (*set_sex)(struct human *p, char sex);
    char (*get_sex)(struct human *p);
} Human; 

void set_age(Human *p, int age) 
{ 
    p->age = age; 
} 

int get_age(Human *p) 
{ 
    return p->age; 
} 

void set_sex(Human *p, char sex)
{ 
    p->sex = sex; 
} 

char get_sex(Human *p) 
{ 
    return p->sex; 
} 

int main() 
{ 
    Human p; 
    p.set_age = set_age; 
    p.set_age(&p, 18); 
    p.set_sex = set_sex; 
    p.set_sex(&p, 'M');
    p.get_age = get_age; 
    printf("age: %d\n", p.get_age(&p));
    p.get_sex = get_sex;  
    printf("sex: %c\n", p.get_sex(&p)); 

    return 0; 
}

我们定义了一个 “human” 的结构体,里面包含了 “age”/“sex” 成员及 “set_age”/“get_age”/”set_sex”/”get_sex” 函数用于实现设置、获取年龄及性别。这就是封装,结构体的数据及函数就能实现 human 的属性及方法操作,另外只有通过结构体中的操作函数才能实现结构体内的数据的更改。

3、继承#

继承就是基于一个已有的类(父类或者基类),再创建一个新的类,这个类被称为子类或者派生类。子类或者派生类可以访问父类的数据及函数,从而避免重复编写代码。子类也可以添加自己的属性和数据。

继承具有如下优点:

  • 提高了代码的复用性,可以避免重复编写相同的代码。
  • 提高了代码的可扩展性,可以支持基于现有类,然后再细微地调整。
  • 提高了代码的可读性,继承使代码更加简洁明了。

在 C 语言里,可以通过结构体嵌套的方式,实现类的继承(这里指的是单继承,暂不考虑多继承),但是需要确保父类结构体引用需要放在子类结构体成员的第一个位置。这样,不论是数据的访问,还是强转都不会有什么问题。

以下便是一个继承的具体例子:

#include <stdio.h>
#include <stdlib.h>

typedef struct human {
    int age;
    char sex;
} Human;

typedef struct person{
    Human human;
    char *name;
} Person;

Person* create_person(int age, char sex, char *name) {
    Person* cperson = (Person*) malloc(sizeof(Person));
    cperson->human.age = age;
    cperson->human.sex = sex;
    cperson->name = name;
    return cperson;
}

int main() {
    Person* cperson = create_person(18, 'w', "lucy");
    printf("(%d, %c) - name: %s\n", cperson->human.age, cperson->human.sex, cperson->name);
    free(cperson);
    return 0;
}

上述代码中,我们定义了两个结构体”Human” 和”Person”。“Person” 包含 “Human” 结构体以及成员变量 “name”。通过函数 “create_person ()” 构造一个 “Person” 类型的结构体,并为其中的 “Human” 成员以及 “name” 成员赋值。当需要在其他地方使用继承的特性时,可以使用类似的嵌套结构体的方式来实现。

4、多态#

多态是面向对象编程中最为核心的概念,它允许我们在不同的对象上执行相同的操作,从而实现灵活的操作处理。

多态具有如下优点:

  • 提高了代码的可扩展性,可以支持不同类型的对象,使程序更加灵活。
  • 提高了代码的可维护性,当类型发生变化时,只需要对相应的类进行修改即可。
  • 提高了代码的可读性,多态使代码更加简洁易读。

C 语言中对于多态的实现,我们可以借助函数指针,利用同一的接口处理不同的数据。具有不同功能的函数可以用同一个函数名,这样可以用一个函数名调用不同功能的函数。

以下便是一个多态的具体例子:

#include <stdio.h>

typedef struct shape {
    void (*draw)(struct shape*);
} Shape;

typedef struct {
    int x;
    int y;
    int radius;
    Shape base;
} Circle;

typedef struct {
    int x1;
    int y1;
    int x2;
    int y2;
    Shape base;
} Line;

void drawCircle(Shape* shape) {
    Circle* circle = (Circle*) shape;
    printf("Circle at (%d, %d) with radius %d\n", circle->x, circle->y, circle->radius);
}

void drawLine(Shape* shape) {
    Line* line = (Line*) shape;
    printf("Line from (%d, %d) to (%d, %d)\n", line->x1, line->y1, line->x2, line->y2);
}

int main() {
    int i; 

    Circle circle = {
        .x = 1,
        .y = 5,
        .radius = 10,
        .base = { .draw = drawCircle }
    };

    Line line = {
        .x1 = 2,
        .y1 = 3,
        .x2 = 7,
        .y2 = 9,
        .base = { .draw = drawLine }
    };
    
    /* //上述两段代码与下面操作是一致的
	Circle circle;
    circle.x = 1;
    circle.y = 5;
    circle.radius = 10;
    circle.base.draw = drawCircle;

    Line line;
    line.x1 = 2;
    line.y1 = 3;
    line.x2 = 7;
    line.y2 = 9;
    line.base.draw = drawLine; 
	*/

    Shape* shapes[2];
    shapes[0] = (Shape*)&(circle.base);
    shapes[1] = (Shape*)&(line.base);

    for (i = 0; i < 2; i++) {
        shapes[i]->draw(shapes[i]);
    }

    return 0;
}

在上面的例子中,我们定义了一个 “Shape” 结构体和它的一个成员函数 “draw”,此外还定义了两个从 “Shape” 派生的子结构体 “Circle” 和 “Line”,并分别实现了它们自己 “draw” 函数。

通过声明让”Shape” 中的”draw” 函数成为了一个函数指针,指向一个参数为”void*” 类型的函数。这让我们可以在”draw” 函数中动态地处理不同的子结构体类型。

我们用”typedef” 将结构体命名为”Shape”,以便在”Circle” 和”Line” 结构体中使用它们。最后,我们在主函数中创建了一个”Circle” 和一个”Line” 对象,并将它们存储在”Shape*” 类型的数组中。使用一个循环来遍历数组,对每个形状对象调用它们自己的”draw” 函数进行处理,这样便实现了多态的效果。

加载中...
此文章数据所有权由区块链加密技术和智能合约保障仅归创作者所有。