c++ char*初始化_C中的继承和多态!是不是找很久了,这里有答案!

06c21a2e65a969cbf1d379713c28fe49.png

1、引言

继承和多态是面向对象语言最强大的功能。有了继承和多态,我们可以完成代码重用。在C中有许多技巧可以实现多态。本文的目的就是演示一种简单和容易的技术,在C中应用继承和多态。通过创建一个VTable(virtual table)和在基类和派生类对象之间提供正确的访问,我们能在C中实现继承和多态。VTable能通过维护一张函数表指针表来实现。为了提供基类和派生类对象之间的访问,我们可以在基类中维护派生类的引用和在派生类中维护基类的引用。

2、说明

在C中实现继承和多态之前,我们应该知道类(Class)在C中如何表示。

2.1、类在C中的表示

考虑C++中的一个类"Person"。

//Person.hclass Person{private:char* pFirstName;char* pLastName; public: Person(constchar* pFirstName,constchar* pLastName);//constructor~Person();//destructorvoiddisplayInfo();voidwriteToFile(constchar* pFileName);};

在C中表示上面的类,我们可以使用结构体,并用操作结构体的函数表示成员函数。

//Person.htypedefstruct_Person{char* pFirstName;char* pLastName;}Person;new_Person(constchar*constpFirstName,constchar*constpLastName);//constructordelete_Person(Person*constpPersonObj);//destructorvoidPerson_DisplayInfo(Person*constpPersonObj);voidPerson_WriteToFile(Person*constpPersonObj,constchar*constpFileName);

这里,定义的操作结构体Person的函数没有封装。为了实现封装,即绑定数据、函数、函数指针。我们需要创建一个函数指针表。构造函数new_Person()将设置函数指针值以指向合适的函数。这个函数指针表将作为对象访问函数的接口。

下面我们重新定义C中实现类Person。

//Person.htypedefstruct_Person Person;//declaration of pointers to functionstypedefvoid(*fptrDisplayInfo)(Person*);typedefvoid(*fptrWriteToFile)( Person*,constchar*);typedefvoid(*fptrDelete)( Person *) ;//Note: In C all the members are by default public. We can achieve //the data hiding (private members), but that method is tricky. //For simplification of this article// we are considering the data members //public only.typedefstruct_Person {char* pFName;char* pLName;//interface for functionfptrDisplayInfo Display; fptrWriteToFile WriteToFile; fptrDelete Delete;}Person;person* new_Person(constchar*constpFirstName,constchar*constpLastName);//constructorvoiddelete_Person(Person*constpPersonObj);//destructorvoidPerson_DisplayInfo(Person*constpPersonObj);voidPerson_WriteToFile(Person*constpPersonObj,constchar* pFileName);

new_Person()函数作为构造函数,它返回新创建的结构体实例。它初始化函数指针接口去访问其它成员函数。这里要注意的一点是,我们仅仅定义了那些允许公共访问的函数指针,并没有给定私有函数的接口。让我们看一下new_Person()函数或C中类Person的构造函数。

//Person.cperson* new_Person(constchar*constpFirstName,constchar*constpLastName){ Person* pObj = NULL;//allocating memorypObj = (Person*)malloc(sizeof(Person));if(pObj == NULL) {returnNULL; } pObj->pFirstName =malloc(sizeof(char)*(strlen(pFirstName)+1));if(pObj->pFirstName == NULL) {returnNULL; }strcpy(pObj->pFirstName, pFirstName); pObj->pLastName =malloc(sizeof(char)*(strlen(pLastName)+1));if(pObj->pLastName == NULL) {returnNULL; }strcpy(pObj->pLastName, pLastName);//Initializing interface for access to functionspObj->Delete = delete_Person; pObj->Display = Person_DisplayInfo; pObj->WriteToFile = Person_WriteToFile;returnpObj;}

创建完对象之后,我们能够访问它的数据成员和函数。

Person* pPersonObj = new_Person("Anjali", "Jaiswal");//displaying person infopPersonObj->Display(pPersonObj);//writing person info in the persondata.txt filepPersonObj->WriteToFile(pPersonObj, "persondata.txt");//delete the person objectpPersonObj->Delete(pPersonObj);pPersonObj = NULL;

注意:不像C++,在C中我们不能在函数中直接访问数据成员。在C++中,可以隐式通过“this”指针直接访问数据成员。我们知道C中是没有“this”指针的,通过显示地传递对象给成员函数。在C中为了访问类的数据成员,我们需要把调用对象作为函数参数传递。上面的例子中,我们把调用对象作为函数的第一个参数,通过这种方法,函数可以访问对象的数据成员。

3、在C中类的表现

Person类的表示——检查初始化接口指向成员函数:

3.1、继承和多态的简单例子

继承-Employee类继承自Person类:

c7bc20b88037b9d54c3659045a241e39.png

在上面的例子中,类Employee继承类Person的属性。因为DisplayInfo()和WriteToFile()函数是virtual的,我们能够从Person的实例访问Employee对象中的同名函数。为了实现这个,我们创建Person实例的时候也初始化Employee类。多态使这成为可能。 在多态的情况下,去解析函数调用,C++使用VTable——即一张函数指针表。

前面我们在结构体中维护的指向函数的指针接口的作用类似于VTable。

//Polymorphism in C++Person PersonObj("Anjali", "Jaiswal");Employee EmployeeObj("Gauri", "Jaiswal", "HR", "TCS", 40000);Person* ptrPersonObj = NULL;//preson pointer pointing to person objectptrPersonObj = &PersonObj;//displaying person infoptrPersonObj ->Display();//writing person info in the persondata.txt fileptrPersonObj ->WriteToFile("persondata.txt");//preson pointer pointing to employee objectptrPersonObj = &EmployeeObj;//displaying employee infoptrPersonObj ->Display();//writing empolyee info in the employeedata.txt fileptrPersonObj ->WriteToFile("employeedata.txt");

在C中,继承可以通过在派生类对象中维护一个基类对象的引用来完成。在基类实例的帮助下,women可以访问基类的数据成员和函数。然而,为了实现多态,街垒对象应该能够访问派生类对象的数据。为了实现这个,基类应该有访问派生类的数据成员的权限。

为了实现虚函数,派生类的函数签名应该和基类的函数指针类似。即派生类函数将以基类对象的一个实例为参数。我们在基类中维护一个派生类的引用。在函数实现上,我们可以从派生类的引用访问实际派生类的数据。

3.2、在C中结构体中的等效表示

C中的继承-Person和Employee结构体:

4938aaf7fcc0a7efc709e134b937878f.png

如图所示,我们在基类结构体中声明了一个指针保存派生类对像,并在派生类结构体中声明一个指针保存基类对象。

在基类对象中,函数指针指向自己的虚函数。在派生类对象的构造函数中,我们需要使基类的接口指向派生类的成员函数。这使我们可以通过基类对象(多态)灵活的调用派生类函数。更多细节,请检查Person和Employee对象的构造函数。

当我们讨论C++中的多态时,有一个对象销毁的问题。为了正确的清楚对象,它使用虚析构函数。在C中,这可以通过使基类的删除函数指针指向派生类的析构函数。派生类的析构函数清楚派生类的数据和基类的数据和对象。注意:检查例子的源码中,实现须构造函数和虚函数的实现细节。

创建Person对象

//Person.htypedefstruct_Person Person;//pointers to functiontypedefvoid(*fptrDisplayInfo)(Person*);typedefvoid(*fptrWriteToFile)(Person*,constchar*);typedefvoid(*fptrDelete)(Person*) ;typedefstruct_person{void* pDerivedObj;char* pFirstName;char* pLastName; fptrDisplayInfo Display; fptrWriteToFile WriteToFile; fptrDelete Delete;}person;Person* new_Person(constchar*constpFristName,constchar*constpLastName);//constructorvoiddelete_Person(Person*constpPersonObj);//destructorvoidPerson_DisplayInfo(Person*constpPersonObj);voidPerson_WriteToFile(Person*constpPersonObj,constchar*constpFileName);//Person.c//construction of Person objectPerson* new_Person(constchar*constpFirstName,constchar*constpLastName){ Person* pObj = NULL;//allocating memorypObj = (Person*)malloc(sizeof(Person));if(pObj == NULL) {returnNULL; }//pointing to itself as we are creating base class objectpObj->pDerivedObj = pObj; pObj->pFirstName =malloc(sizeof(char)*(strlen(pFirstName)+1));if(pObj->pFirstName == NULL) {returnNULL; }strcpy(pObj->pFirstName, pFirstName); pObj->pLastName =malloc(sizeof(char)*(strlen(pLastName)+1));if(pObj->pLastName == NULL) {returnNULL; }strcpy(pObj->pLastName, pLastName);//Initializing interface for access to functions//destructor pointing to destrutor of itselfpObj->Delete = delete_Person; pObj->Display = Person_DisplayInfo; pObj->WriteToFile = Person_WriteToFile;returnpObj;}

Person对象的结构

2b59907dcf43ca425837ec938b3fb348.png

创建Employee对象

//Employee.h#include "Person.h"typedefstruct_Employee Employee;//Note: interface for this class is in the base class//object since all functions are virtual.//If there is any additional functions in employee add//interface for those functions in this structure typedefstruct_Employee{ Person* pBaseObj;char* pDepartment;char* pCompany;intnSalary;//If there is any employee specific functions; add interface here.}Employee;Person* new_Employee(constchar*constpFirstName,constchar*constpLastName,constchar*constpDepartment,constchar*constpCompany,intnSalary);//constructorvoiddelete_Employee(Person*constpPersonObj);//destructorvoidEmployee_DisplayInfo(Person*constpPersonObj);voidEmployee_WriteToFile(Person*constpPersonObj,constchar*constpFileName);//Employee.cPerson* new_Employee(constchar*constpFirstName,constchar*constpLastName,constchar*constpDepartment,constchar*constpCompany,intnSalary){ Employee* pEmpObj;//calling base class construtorPerson* pObj = new_Person(pFirstName, pLastName);//allocating memorypEmpObj =malloc(sizeof(Employee));if(pEmpObj == NULL) { pObj->Delete(pObj);returnNULL; } pObj->pDerivedObj = pEmpObj;//pointing to derived object//initialising derived class memberspEmpObj->pDepartment =malloc(sizeof(char)*(strlen(pDepartment)+1));if(pEmpObj->pDepartment == NULL) {returnNULL; }strcpy(pEmpObj->pDepartment, pDepartment); pEmpObj->pCompany =malloc(sizeof(char)*(strlen(pCompany)+1));if(pEmpObj->pCompany== NULL) {returnNULL; }strcpy(pEmpObj->pCompany, pCompany); pEmpObj->nSalary = nSalary;//Changing base class interface to access derived class functions//virtual destructor//person destructor pointing to destrutor of employeepObj->Delete = delete_Employee; pObj->Display = Employee_DisplayInfo; pObj->WriteToFile = Employee_WriteToFile;returnpObj;}

Employee对象的结构

bd3032c740ef086906f642eaea905244.png

注意:从基类函数到派生类函数改变了接口(VTable)中指针位置。现在我们可以从基类(多态)访问派生类函数。我们来看如何使用多态。

Person* PersonObj = new_Person("Anjali", "Jaiswal");Person* EmployeeObj = new_Employee("Gauri", "Jaiswal","HR", "TCS", 40000);//accessing person object//displaying person infoPersonObj->Display(PersonObj);//writing person info in the persondata.txt filePersonObj->WriteToFile(PersonObj,"persondata.txt");//calling destructorPersonObj->Delete(PersonObj);//accessing to employee object//displaying employee infoEmployeeObj->Display(EmployeeObj);//writing empolyee info in the employeedata.txt fileEmployeeObj->WriteToFile(EmployeeObj, "employeedata.txt");//calling destrutorEmployeeObj->Delete(EmployeeObj);

结论

使用上面描述的简单的额外代码能是过程式C语言有多态和继承的特性。我们简单的使用函数指针创建一个VTable和在基类和派生类对象中交叉维护引用。用这些简单的步骤,我们在C中可以实现继承和多态。

乎:如果你也想成为一名程序员那就关注我与我交流,不管是零基础还是入门小白学习从来不是一个人的事情,要有个相互监督的伙伴,工作需要学习C/C++或者感兴趣、为了入行、转行学习C/C++的伙伴可以一起学习!关注小编的专栏,每晚技术大牛手把手教你如何实现!

程序猿​zhuanlan.zhihu.com
bbb23230d396133867ae0d6fe7f23100.png

作者:吴秦

出处:http://www.cnblogs.com/skynet/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值