C语言实现多态

首先声明,大神就不要看了。小弟水平有限。

C++多态是通过虚函数表实现的,类似于JAVA多态的实现方式。关于Java多态的实现方式可以看我之前写过的一篇不是很完善的文章。从JVM角度看Java多态

Java和C++不同,Java中所有的实例方法(相对于类方法,或叫静态方法而言)都是默认为虚函数,之前貌似看到过Java生成的字节码中,所有实例方法前面都是右virtual关键字的。C++中需要显示声明virtual之后才是虚函数,虚函数是实现多态的基础。

今天用C语言实现的多态,是实现一个类似下面的C++代码:(由于使用QtCreator写的,所以会有一点儿QT的代码,可以忽略)

#include <QCoreApplication>
#include <iostream>
using namespace std; class Base{
public:
virtual void eat(){
cout<<"基类在吃饭....."<<endl;
}
virtual void play(){
cout<<"基类在玩耍....."<<endl;
}
}; class DeriveA:public Base{
public:
void eat(){
cout<<"子类A在吃饭....."<<endl;
}
};
class DeriveB:public Base{
public:
void eat(){
cout<<"子类B在吃饭....."<<endl;
}
}; int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Base * base;
DeriveA dA;
DeriveB dB;
base = &dA;
base->eat();
base->play();
cout<<"---------------------------------------------\n";
base = &dB;
base->eat();
base->play(); return a.exec();
}

其中,基类中有两个虚函数,eat()和play(),两个派生类中都只重写了基类的eat()方法。

输出结果如下图:

下面用纯C语言实现类似的效果。可以用C语言模仿C++的虚函数表。

首先定义两个函数指针类型:

typedef void (*EatPtr)();
typedef void (*PlayPtr)();

接着模拟虚函数表,虚函数表就是一个元素为虚函数指针的结构体

typedef struct _virtualPtrTable{
EatPtr eat;
PlayPtr play;
}VPtrTable;

接着定义“基类和派生类”:

typedef struct _base{
VPtrTable vptrTable;
int age;
}Base; typedef struct _deriveA{
Base base;
int age;
}DeriveA; typedef struct _deriveB{
Base base;
int age;
}DeriveB;

接着实现函数,由于C++代码中,两个派生类都没有实现play方法,所以这里派生类的play函数都只是调用基类的函数。。

/** 派生类A的实现函数 **/
void aEat(){
cout<<"子类A在吃饭....."<<endl;
}
void aPlay(){
basePlay();
} /** 派生类B的实现函数 **/
void bEat(){
cout<<"子类B在吃饭....."<<endl;
}
void bPlay(){
basePlay();
}

下面是主函数:

int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Base *base;
DeriveA deriveA;
deriveA.base.vptrTable.eat = aEat;
deriveA.base.vptrTable.play = aPlay;
deriveA.base.age = ;
deriveA.age = ; DeriveB deriveB;
deriveB.base.vptrTable.eat = bEat;
deriveB.base.vptrTable.play = bPlay;
deriveB.base.age = ;
deriveB.age = ; base = (Base *)&deriveA;
base->vptrTable.eat();
base->vptrTable.play();
cout<<"age:"<<base->age<<endl;
cout<<"---------------------------------------------\n";
base = (Base *)&deriveB;
base->vptrTable.eat();
base->vptrTable.play();
cout<<"age:"<<base->age<<endl; return a.exec();
}

完整代码如下:

#include <QCoreApplication>
#include <iostream>
using namespace std; typedef void (*EatPtr)();
typedef void (*PlayPtr)(); typedef struct _virtualPtrTable{
EatPtr eat;
PlayPtr play;
}VPtrTable; typedef struct _base{
VPtrTable vptrTable;
int age;
}Base; typedef struct _deriveA{
Base base;
int age;
}DeriveA; typedef struct _deriveB{
Base base;
int age;
}DeriveB; /** 基类的实现函数 **/
void baseEat(){
cout<<"基类在吃饭....."<<endl;
}
void basePlay(){
cout<<"基类在玩耍....."<<endl;
} /** 派生类A的实现函数 **/
void aEat(){
cout<<"子类A在吃饭....."<<endl;
}
void aPlay(){
basePlay();
} /** 派生类B的实现函数 **/
void bEat(){
cout<<"子类B在吃饭....."<<endl;
}
void bPlay(){
basePlay();
} int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Base *base;
DeriveA deriveA;
deriveA.base.vptrTable.eat = aEat;
deriveA.base.vptrTable.play = aPlay;
deriveA.base.age = ;
deriveA.age = ; DeriveB deriveB;
deriveB.base.vptrTable.eat = bEat;
deriveB.base.vptrTable.play = bPlay;
deriveB.base.age = ;
deriveB.age = ; base = (Base *)&deriveA;
base->vptrTable.eat();
base->vptrTable.play();
cout<<"age:"<<base->age<<endl;
cout<<"---------------------------------------------\n";
base = (Base *)&deriveB;
base->vptrTable.eat();
base->vptrTable.play();
cout<<"age:"<<base->age<<endl; return a.exec();
}

运行效果:

写的比较简单,欢迎来探讨。

如果你觉得有所收获,记得点赞呀~~

C语言实现多态的更多相关文章

  1. C 语言实现多态的原理:函数指针

    C语言实现多态的原理:函数指针 何为函数指针?答案:C Programming Language. 能够查阅下,从原理上来讲,就是一个内存地址.跳过去运行相应的代码段. 既然如此,在运行时决定跳到哪个 ...

  2. python语言的鸭子类型和强类型语言的多态

    python语言的鸭子类型和强类型语言的多态 前面讲接口类的时候举过一个有关支付方式的例子,支付方式可以有几种,微信支付,支付宝支付,苹果支付等,这几个不同的支付都统一于支付,像这样几个类都统一于 某 ...

  3. c#基础语言编程-多态

    语言中的多态性是为了使程序有扩展性,为实现多态性,在程序中体现为接口.抽象类.父类.具体类. 接口就是一种规范,解决了多重继承的问题,类似一种规范,告诉我要做什么,具有什么能力,在接口中定义写行为属性 ...

  4. [Objective-C语言教程]多态(26)

    多态性这个词表示有许多形式. 通常,当存在类的层次结构并且通过继承相关时,会发生多态性. Objective-C多态表示对成员函数的调用将导致执行不同的函数,具体取决于调用该函数的对象的类型. 考虑下 ...

  5. 面试问题之C++语言:多态

    什么是多态? 概念:同一操作作用于不同的对象,可以有不同的解释,产生不同的执行结果,这就是多态性.简单的说,就是用基类的引用指向子类的对象. 为什么要用多态呢? 原因:封装可以隐藏实现细节,使得代码模 ...

  6. 关于c语言模拟c++的多态

    关于c++多态,个人认为就是父类调用子类的方法,c++多态的实现主要通过虚函数实现,如果类中含有虚函数,就会出现虚函数表,具体c++多态可以参考<深度探索c++对象模型> c语言模拟多态主 ...

  7. c语言实现封装、继承和多态

    1.  概述 C语言是一种面向过程的程序设计语言,而C++是在C语言基础上衍生来了的面向对象的语言,实际上,很多C++实现的底层是用C语言实现的,如在Visual C++中的Interface其实就是 ...

  8. Python源代码 -- C语言实现面向对象编程(基类&amp;派生类&amp;多态)

    背景 python是面向对象的解释性语言.然而python是通过C语言实现的,C语言怎么跟面向对象扯上了关系? C语言能够实现面向对象的性质? 原文链接:http://blog.csdn.net/or ...

  9. Go语言多态

    总结一下Go语言中多态 package main import "fmt" //申明一个函数类型 type FuncMs func(int ,int) int //加法 func ...

随机推荐

  1. 线程的使用方法start run sleep join

    今天回顾了Java的线程的一些知识 例1:下面代码存有详细的解释 主要是继承Thread类与实现Runnable接口 以及start()和run()方法 package com.date0607; / ...

  2. django配合mongo使用

    环境 django 1.11.16 mongoengine 0.16.0 需要安装mongoengine库 pip install mongoengine 1.在配置文件中 # settings.py ...

  3. dpkg: error: dpkg status database is locked by another process

    First run: lsof /var/lib/dpkg/lock Then make sure that process isn't running: ps cax | grep PID If i ...

  4. AutoCAD开发2--添加带属性的点

    Private Sub CommandButton11_Click() Dim pPoint As AcadPoint Dim DataType(0 To 1) As Integer Dim Data ...

  5. squid常用操作

    如何查看squid的缓存命中率 使用命令: squidclient -h host -p port mgr:info比如: /usr/local/squid/bin/squidclient -h 12 ...

  6. MFC在对话框中嵌入对话框

    在对话框中嵌入子对话框 代码 m_childDlg = new CChildDlg(); m_childDlg->Create(IDD_CHILD_DIALOG,AfxGetApp()-> ...

  7. C++与C语言在结构体上的区别

    用Nios 实现逻辑上很清楚,只是C++用switch语句后,写的很麻烦,主要是Switch语句很长吧. 另外要记录下:struct在C++中,在a文件中定义在b文件中定义变量是可以的,但在C语言中, ...

  8. Reading | 《C++ Primer Plus》(未完待续)

    目录 一.概述和C++简史 1.早期语言的问题 2.面向对象编程OOP 3.泛型编程 二.入门 1.头文件 2.名称空间 3.cout输出 4.C++语句 5.函数 一.概述和C++简史 C++融合了 ...

  9. Hadoop 系列文章(二) Hadoop配置部署启动HDFS及本地模式运行MapReduce

    接着上一篇文章,继续我们 hadoop 的入门案例. 1. 修改 core-site.xml 文件 [bamboo@hadoop-senior hadoop-2.5.0]$ vim etc/hadoo ...

  10. 【洛谷4070】 [SDOI2016]生成魔咒(SAM)

    传送门 洛谷 Solution 考虑要求的是什么,前缀的本质不同的字符串个数? 如果只要求一个串那么显然答案是\(\sum_{i=1}^{tot}len[i]-len[fa[i]]\)(实际上这个并不 ...