diff --git a/README.md b/README.md index 67357dd..085b490 100644 --- a/README.md +++ b/README.md @@ -20,6 +20,8 @@ - [x] [extern那些事](./extern) - [x] [struct那些事](./struct) - [x] [struct与class那些事](./struct_class) +- [x] [union那些事](./union) +- [x] [c实现c++多态](./c_poly) 代码运行: 全部在linux下用vim编写,使用gcc/g++调试!全部可正常运行! diff --git a/c_poly/README.md b/c_poly/README.md new file mode 100644 index 0000000..0b2ea41 --- /dev/null +++ b/c_poly/README.md @@ -0,0 +1,32 @@ +# C实现C++的面向对象特性 +## 1.C++实现案例 + +C++中的多态:在C++中会维护一张虚函数表,根据赋值兼容规则,我们知道父类的指针或者引用是可以指向子类对象的。 + +如果一个父类的指针或者引用调用父类的虚函数则该父类的指针会在自己的虚函数表中查找自己的函数地址,如果该父类对象的指针或者引用指向的是子类的对象,而且该子类已经重写了父类的虚函数,则该指针会调用子类的已经重写的虚函数。 + +学习案例代码见:[c++_examp.cpp](./c++_examp.cpp) + + + +## 2.C实现 + +- 封装 + +C语言中是没有class类这个概念的,但是有struct结构体,我们可以考虑使用struct来模拟; + +使用函数指针把属性与方法封装到结构体中。 + +- 继承 + +结构体嵌套 + +- 多态 + +类与子类方法的函数指针不同 + +在C语言的结构体内部是没有成员函数的,如果实现这个父结构体和子结构体共有的函数呢?我们可以考虑使用函数指针来模拟。但是这样处理存在一个缺陷就是:父子各自的函数指针之间指向的不是类似C++中维护的虚函数表而是一块物理内存,如果模拟的函数过多的话就会不容易维护了。 + + + +学习案例代码见:[c_examp.c](./c_examp.c) diff --git a/c_poly/c++_examp b/c_poly/c++_examp new file mode 100755 index 0000000..fe986fd Binary files /dev/null and b/c_poly/c++_examp differ diff --git a/c_poly/c++_examp.cpp b/c_poly/c++_examp.cpp new file mode 100644 index 0000000..3cb6bc6 --- /dev/null +++ b/c_poly/c++_examp.cpp @@ -0,0 +1,42 @@ +/** + * @file c++_examp.cpp + * @brief c++中的多态 + * @author 光城 + * @version v1 + * @date 2019-08-06 + */ + +#include + +using namespace std; +class A +{ + public: + virtual void f()//虚函数实现 + { + cout << "Base A::f() " << endl; + } +}; + +class B:public A // 必须为共有继承,否则后面调用不到,class默认为私有继承! +{ + public: + virtual void f()//虚函数实现,子类中virtual关键字可以没有 + { + cout << "Derived B::f() " << endl; + } +}; + + +int main() +{ + A a;//基类对象 + B b;//派生类对象 + + A* pa = &a;//父类指针指向父类对象 + pa->f();//调用父类的函数 + + pa = &b; //父类指针指向子类对象,多态实现 + pa->f();//调用派生类同名函数 + return 0; +} diff --git a/c_poly/c_examp b/c_poly/c_examp new file mode 100755 index 0000000..458fb2a Binary files /dev/null and b/c_poly/c_examp differ diff --git a/c_poly/c_examp.c b/c_poly/c_examp.c new file mode 100644 index 0000000..a083db8 --- /dev/null +++ b/c_poly/c_examp.c @@ -0,0 +1,55 @@ +/** + * @file c_examp.c + * @brief C实现多态 + * @author 光城 + * @version v1 + * @date 2019-08-06 + */ + +#include + +/// 重定义一个函数指针类型 +typedef void (*pf) (); + +/** + * @brief 父类 + */ +typedef struct _A +{ + pf _f; +}A; + + +/** + * @brief 子类 + */ +typedef struct _B +{ + A _b; ///< 在子类中定义一个基类的对象即可实现对父类的继承。 +}B; + +void FunA() +{ + printf("%s\n","Base A::fun()"); +} + +void FunB() +{ + printf("%s\n","Derived B::fun()"); +} + + +int main() +{ + A a; + B b; + + a._f = FunA; + b._b._f = FunB; + + A *pa = &a; + pa->_f(); + pa = (A *)&b; /// 让父类指针指向子类的对象,由于类型不匹配所以要进行强转 + pa->_f(); + return 0; +} diff --git a/union/README.md b/union/README.md new file mode 100644 index 0000000..5811fb9 --- /dev/null +++ b/union/README.md @@ -0,0 +1,12 @@ +# UNION那些事 + +联合(union)是一种节省空间的特殊的类,一个 union 可以有多个数据成员,但是在任意时刻只有一个数据成员可以有值。当某个成员被赋值后其他成员变为未定义状态。联合有如下特点: + +- 默认访问控制符为 public +- 可以含有构造函数、析构函数 +- 不能含有引用类型的成员 +- 不能继承自其他类,不能作为基类 +- 不能含有虚函数 +- 匿名 union 在定义所在作用域可直接访问 union 成员 +- 匿名 union 不能包含 protected 成员或 private 成员 +- 全局匿名联合必须是静态(static)的 diff --git a/union/union b/union/union new file mode 100755 index 0000000..36082ab Binary files /dev/null and b/union/union differ diff --git a/union/union.cpp b/union/union.cpp new file mode 100644 index 0000000..7439282 --- /dev/null +++ b/union/union.cpp @@ -0,0 +1,50 @@ +/** + * @file union.cpp + * @brief UNION + * @author 光城 + * @version v1 + * @date 2019-08-06 + */ + +#include +/** + * 默认访问控制符为public + */ +union UnionTest { + /** + * 可以含有构造函数、析构函数 + */ + UnionTest() : i(10) {print(i);}; + ~UnionTest(){}; + int i; +private: + void print(int i){std::cout<