C++核心编程

朴灿烈づ我的快乐病毒、 2022-12-13 01:55 264阅读 0赞

13 C++类型转换

C风格转换【(TYPE)EXPRESSION】可在任意类型之间转换,且不易查找。所以C++引进了四种类型转换操作符,解决以上问题。


























类型 主要用途
static_cast 静态类型转换
dynamic_cast 子类和父类间多态类型转换
const_cast 去掉const属性转换
reinterpreter_cast 重新解释类型转换

13.1 static_cast

  • static_cast<目标类型>(标识符)
  • 静态,即在编译期内就可完成其类型转换,最经常使用

    include

    int main(){

    1. double pi = 3.1415926;
    2. int num1 = (int)pi; // C语言旧式类型转换
    3. int num2 = pi; // 隐式类型转换
    4. //C++静态类型转换
    5. //在编译期完成基本类型的转换,且有类型检查
    6. int num3 = static_cast<int> (pi);
    7. std::cout << "num1 = " << num1 << std::endl;
    8. std::cout << "num2 = " << num2 << std::endl;
    9. std::cout << "num3 = " << num3 << std::endl;
    10. system("pause");
    11. return 0;

    }

13.2 dynamic_cast

  • dynamic_cast<目标类型>(标识符)
  • 用于多态中父子类之间的多态转换,转换类指针时,基类需要虚函数

    include

    class Animal{
    public:

    1. virtual void shout() = 0;

    };

    class Dog : public Animal
    {
    public:

    1. virtual void shout(){
    2. std::cout << "汪汪" << std::endl;
    3. }
    4. void dohome()
    5. {
    6. std::cout << "看家" << std::endl;
    7. }

    };

    class Cat : public Animal
    {
    public:

    1. virtual void shout(){
    2. std::cout << "喵喵" << std::endl;
    3. }
    4. void dohome()
    5. {
    6. std::cout << "抓老鼠" << std::endl;
    7. }

    };

    int main(){

    1. Animal* base = NULL;
    2. base = new Cat();
    3. base->shout();
    4. //将父类指针转成子类
    5. //此时转换失败,因为父类指针现在指向的对象是猫
    6. //转换失败返回空(NULL)
    7. Dog* dog = dynamic_cast<Dog*> (base);
    8. if (NULL!=dog)
    9. {
    10. dog->shout();
    11. dog->dohome();
    12. }
    13. //此时转换成功,成功将父类指针转换成子类指针
    14. Cat* cat = dynamic_cast<Cat*> (base);
    15. if (cat != NULL){
    16. cat->shout();
    17. cat->dohome();
    18. }
    19. system("pause");
    20. return 0;

    }

13.3 const_cast

const_cast<目标类型>(标识符):目标类型只能是指针或者引用

  1. #include<iostream>
  2. class A
  3. {
  4. public:
  5. int data;
  6. };
  7. int main{
  8. //类似于结构体的初始化方式
  9. const A a = { 200 };
  10. //编译失败:const_cast 目标类型只能是引用或者指针
  11. //A a1 = const_cast<A> (a);
  12. A& a2 = const_cast<A&> (a);
  13. a2.data = 100;
  14. std::cout << &a << "——" << a.data << std::endl;
  15. std::cout << &a2 << "——" << a2.data << std::endl;
  16. A* a3 = const_cast<A*> (&a2);
  17. a3->data = 100;
  18. std::cout << &a3 << "——" << a3->data << std::endl;
  19. system("pause");
  20. return 0;
  21. }

扩展:取址符号的解释

  1. #include<iostream>
  2. //& 表示取址符号,取的是当前变量本身的内存地址
  3. int a = 1; // 变量a
  4. int &c = a; // 引用c
  5. int *b = &a; // 指针b
  6. int main()
  7. {
  8. std::cout << &c << std::endl; //输出的是c的存储地址也就是a的地址
  9. std::cout << *&c << std::endl; //*取这个地址的值,输出1
  10. std::cout << b << std::endl; //输出指针指向的地址,即a的地址
  11. std::cout << &b << std::endl; //取的是指针b本身的地址
  12. system("pause");
  13. return 0;
  14. }

13.4 reinterpret_cast

  • reinterpret_cast<目标类型>(标识符)
  • 数据的二进制重新解释,但是不改变其值

reinterpret_cast是四种强制转换中功能最强大的,它可以暴力完成两个完全无关类型的指针之间或指针和数之间的互转,比如用char类型指针指向double值。它对原始对象的位模式提供较低层次上的重新解释(即reinterpret),完全复制二进制比特位到目标对象,转换后的值与原始对象无关但比特位一致,前后无精度损失

指针和数之间的互转

  1. #include<iostream>
  2. int main(){
  3. double d = 12.1;
  4. char* p = reinterpret_cast<char*>(&d);
  5. std::cout << "*p的值:" << *p << std::endl;
  6. // 将d以二进制(位模式)方式解释为char,并赋给*p
  7. double* q = reinterpret_cast<double*>(p);
  8. std::cout << "*q的值:" << *q << std::endl; // 12.1
  9. }

无关类型指针之间的类型转换

  1. #include<iostream>
  2. class Person{
  3. public:
  4. void run(){
  5. std::cout << "我在不停的奔跑" << std::endl;
  6. }
  7. };
  8. class Sky{
  9. public:
  10. void laugh(){
  11. std::cout << "天空乌云密布,暴雨要来了" << std::endl;
  12. }
  13. };
  14. int main(){
  15. Person* p = new Person();
  16. p->run();
  17. Sky* sky = reinterpret_cast<Sky*> (p);
  18. sky->laugh();
  19. std::cout << &p << "——" << std::endl;
  20. std::cout << &sky << "——" << std::endl;
  21. system("pause");
  22. return 0;
  23. }

14 C++内存知识

14.1 内存区域划分

在C++中,程序在内存中的存储被分为五个区:

1、代码区(Text):只读,共享,静态分配,存放程序执行代码,也可能包含常量

2、全局区:编译时分配内存,程序结束后由系统释放,存放全局变量和静态变量

  • DATA: 也叫GVAR(global value),存放已初始化的非零全局变量
  • BSS:存放零值或未初始化的全局变量,在程序开始时被清零

3、栈区(stack):由编译器自动分配释放 ,存放函数参数值,局部变量值等,按内存地址由高到低方向生长,其空间大小由编译时确定,速度快,空间小

4、堆区(heap):由程序员分配释放,按内存地址由低到高方向生长,其空间大小由系统内存上限决定,速度慢,空间大

每个线程都会有自己的栈,但是堆空间是共用的,下图为C++内存区域划分图

在这里插入图片描述

14.2 列举分配空间场景


























区域类型 列举场景
代码区 类的成员函数和全局函数
全局区 全局变量和静态变量
栈区 局部变量和函数参数值
堆区 调用malloc函数或new运算符重载申请空间

14.3 变量的生命周期




































类型 内存划分 作用范围 生命周期
局部变量 定义局部变量所在的大括号内 定义的函数被调用时,在栈开辟空间,函数调用结束,被清除
全局变量 全局区 所有源文件 在编译期即分配好了内存空间,在程序运行结束时,被清除
静态局部变量 全局区 定义局部变量所在的大括号内 在编译期即分配好了内存空间,在程序运行结束时,被清除
静态全局变量 全局区 只能在定义的文件中调用 在编译期即分配好了内存空间,在程序运行结束时,被清除

14.4 名称空间的用途

避免变量或函数重命名,命名空间可以将多个变量和函数等包含在内,使其不会与命名空间外的任何变量和函数等发生重命名的冲突

  1. namespace ride{
  2. void PrintNum(int x, int y){
  3. std::cout << x*y << std::endl;
  4. }
  5. }//打印两数相乘的命名空间
  6. namespace plus{
  7. void PrintNum(int x, int y){
  8. std::cout << x+y << std::endl;
  9. }
  10. }//打印两数相加的命名空间
  11. int main()
  12. {
  13. ride::PrintNum(1, 4);
  14. plus::PrintNum(1, 4);
  15. system("pause");
  16. return 0;
  17. }

14.5 运行Test函数

例一

  1. void GetMemory(char *p)
  2. {
  3. p = (char *)malloc(100);
  4. }
  5. void Test(void)
  6. {
  7. char *str = NULL;
  8. GetMemory(str);
  9. strcpy(str, "hello world");
  10. printf(str);
  11. }

结果:编译成功,运行失败

原因:因为调用GetMemory相当于值传递,str指针实际上并没有分配到内存空间,访问空指针会运行失败,因为在C++中内存编号0 ~255为系统占用内存,内存地址编号为0的空间不允许用户访问

详细过程:在函数地址入栈后,p作为形参,只是拷贝了一份str的地址,并为该地址申请了空间,但是str指向的地址仍然是NULL

解决方法一

  1. void GetMemory(char **p)
  2. {
  3. *p = (char *)malloc(100);
  4. }
  5. void Test(void)
  6. {
  7. char *str = NULL;
  8. GetMemory(&str);
  9. strcpy(str, "hello world");
  10. printf(str);
  11. }

解决方法二

  1. void GetMemory(char* &p)
  2. {
  3. p = (char *)malloc(100);
  4. }
  5. void Test(void)
  6. {
  7. char *str = NULL;
  8. GetMemory(str);
  9. strcpy(str, "hello world");
  10. printf(str);
  11. }

例二

  1. char *GetMemory(void)
  2. {
  3. char p[] = "hello world";
  4. return p;
  5. }
  6. void Test(void)
  7. {
  8. char *str = NULL;
  9. str = GetMemory();
  10. printf(str);
  11. }

结果:打印结果不确定

原因:调用函数返回值是局部变量的指针,局部变量在函数被调用结束后,在栈区的内存空间被释放。因此str在打印时,指针指向的是非法(不可控)的内存地址,即野指针,在C++中访问野指针的结果是不确定的

例三

  1. void GetMemory2(char **p, int num)
  2. {
  3. *p = (char *)malloc(num);
  4. }
  5. void Test(void)
  6. {
  7. char *str = NULL;
  8. GetMemory(&str, 100);
  9. strcpy(str, "hello");
  10. printf(str);
  11. }

结果:正常打印hello

原因:地址传递,把str指针的地址当做实参传递给形参,并在GetMemory2中为该地址申请地址空间

例四

  1. void Test(void)
  2. {
  3. char *str = (char *) malloc(100);
  4. strcpy(str, hello”);
  5. free(str);
  6. if(str != NULL)
  7. {
  8. strcpy(str, world”);
  9. printf(str);
  10. }
  11. }

结果:运行时中断,后打印world

原因:free只是释放了malloc所申请的内存,并没有改变指针的值,因此由于指针所指向的内存已经被释放,所以其它代码有机会改写其内容,相当于该指针指向了自己无法控制的地方,也称为野指针。为了避免操作野指针,在free后应将指针指向NULL

15 函数的使用

15.1 函数指针

函数指针:指向函数的指针,可通过指针访问函数,函数名作为实参使用时

作用:通过使用函数指针可以简化代码,在一定程度上提高代码可读性

  1. #include<iostream>
  2. int add(int a, int b){
  3. return a + b;
  4. }
  5. typedef int(*FucPoint)(int, int);
  6. //使用函数指针作为形参
  7. void myFun(int a, int b, FucPoint p){
  8. p(a, b);
  9. }
  10. int main(){
  11. //函数名作为入参时,C++会自动将其转换为函数指针
  12. myFun(2, 3, add);
  13. system("pause");
  14. return 0;
  15. }

15.2 静态函数

静态函数:被static关键字修饰的函数,静态成员函数属于类,不创建对象也可调用。类成员函数与静态成员函数的区别在于普通成员函数有一个隐藏的调用参数(this)指针

应用场景:从面向对象的角度上来说,在抉择使用实例化方法或静态方法时,应该根据是否该方法和实例化对象具有逻辑上的相关性,如果是就应该使用实例化对象,反之使用静态方法

15.3 编程实现冒泡

要求:

  1. 创建一个无序的数组,通过定义的排序算法函数sort排序并输出
  2. 排序算法函数sort使用排序规则回调函数(函数指针)作为参数传入
  1. #include<iostream>
  2. bool judge(int a, int b){
  3. return a>b?true:false;
  4. }
  5. typedef bool(*FucPoint)(int, int);
  6. class SortUtil{
  7. public:
  8. static int* sort(int arr[], int len, FucPoint p){
  9. for (int i = 0; i < len - 1; ++i)
  10. {
  11. for (int j = 0; j < len - 1 - i; ++j)
  12. {
  13. if (p(arr[j] , arr[j + 1]))
  14. {
  15. int temp = arr[j];
  16. arr[j] = arr[j + 1];
  17. arr[j + 1] = temp;
  18. }
  19. }
  20. }
  21. return arr;
  22. }
  23. };
  24. int main() {
  25. int arr[9] = { 4, 2, 8, 0, 5, 7, 1, 3, 9 };
  26. int len = sizeof(arr) / sizeof(int);
  27. SortUtil::sort(arr,len,judge);
  28. system("pause");
  29. return 0;
  30. }

16 类的封装

16.1构造一个类

要求:实现类的无参构造、有参构造、拷贝构造、拷贝复制、析构函数

  1. #include<iostream>
  2. class Person {
  3. public:
  4. Person() {
  5. }
  6. Person(int age, int height) {
  7. age_ = age;
  8. height_ = new int(height);
  9. }
  10. Person(const Person& p) {
  11. age_ = p.age_;
  12. height_ = new int(*p.height_);
  13. }
  14. virtual ~Person() {
  15. if (height_ != NULL)
  16. {
  17. delete height_;
  18. }
  19. }
  20. private:
  21. int age_;
  22. int* height_;
  23. };

扩展:构造函数初始化列表与构造函数中的赋值的区别

测试代码如下

  1. #include<iostream>
  2. class A{
  3. public:
  4. A(){
  5. std::cout << "无参构造" << std::endl;
  6. }
  7. A(const A& a){
  8. std::cout << "拷贝构造" << std::endl;
  9. }
  10. A& operator = (const A &a){
  11. std::cout << "运算符重载" << std::endl;
  12. return *this;
  13. }
  14. };
  15. class B{
  16. public:
  17. B(const A& val) :m_b_(val){
  18. std::cout << "参数列表初始化" << std::endl;
  19. }
  20. private:
  21. A m_b_;
  22. };
  23. class C {
  24. public:
  25. C(const A& c) {
  26. m_c_ = c;
  27. std::cout << "构造函数内赋值" << std::endl;
  28. }
  29. private:
  30. A m_c_;
  31. };
  32. int main(){
  33. A a;
  34. system("pause");
  35. B b(a);
  36. system("pause");
  37. C c(a);
  38. system("pause");
  39. return 0;
  40. }

程序执行结果如图所示:
在这里插入图片描述

执行B b(a) ,直接调用A(const A& a)拷贝构造函数完成初始化

执行C c(a) ,先调用A对象的无参构造函数,后调用A对象的重载运算符函数

A& operator = (A &a)

结论:不同的初始化方式得到不同的结果,构造函数内赋值初始化多调用了一次构造函数,因此构造函数初始化列表的方式效率更高

16.2 编写圆形程序

定义一个圆形类Circle

要求:包含 私有成员(半径),公有函数成员(设置半径、计算面积、计算周长),构造函数(无参构造、有参数构造、拷贝构造)

主函数使用圆形类Circle创建圆形对象

  1. 创建圆对象c1,键盘输入x,设定为c1半径,计算并显示c1面积和周长
  2. 创建圆对象c2,初始化半径为2x,计算并显示c2面积和周长
  3. 创建圆对象c3,并用c1初始化c3,计算并显示c3面积和周长

Math.h中有下面一段:

  1. #if defined(_USE_MATH_DEFINES) && !defined(_MATH_DEFINES_DEFINED)
  2. #define _MATH_DEFINES_DEFINED
  3. #define M_E 2.71828182845904523536
  4. #define M_LOG2E 1.44269504088896340736
  5. #define M_LOG10E 0.434294481903251827651
  6. #define M_LN2 0.693147180559945309417
  7. #define M_LN10 2.30258509299404568402
  8. #define M_PI 3.14159265358979323846
  9. #define M_PI_2 1.57079632679489661923
  10. #define M_PI_4 0.785398163397448309616
  11. #define M_1_PI 0.318309886183790671538
  12. #define M_2_PI 0.636619772367581343076
  13. #define M_2_SQRTPI 1.12837916709551257390
  14. #define M_SQRT2 1.41421356237309504880
  15. #define M_SQRT1_2 0.707106781186547524401
  16. #endif /* _USE_MATH_DEFINES */

圆形类代码实现

  1. #include<iostream>
  2. #define _USE_MATH_DEFINES
  3. #include <math.h>
  4. class Circle{
  5. public:
  6. //被const修饰的成员函数不能引用非const函数,也不能修改类内的变量值,非常安全
  7. void setRadius(const double r){
  8. radius_ = r;
  9. }
  10. double getPerimeter() const{
  11. return 2 * M_PI*radius_;
  12. }
  13. double getArea() const{
  14. return M_PI*radius_*radius_;
  15. }
  16. Circle() {
  17. }
  18. explicit Circle(double r) {
  19. radius_ = r;
  20. }
  21. Circle(const Circle& c) {
  22. radius_ = c.radius_;
  23. }
  24. private:
  25. double radius_;
  26. };
  27. int main() {
  28. Circle c1;
  29. double x;
  30. std::cout << "请输入圆形的半径" << std::endl;
  31. std::cin >> x;
  32. c1.setRadius(x);
  33. double c1_perimeter = c1.getPerimeter();
  34. double c1_area = c1.getArea();
  35. std::cout << "c1的周长为:" << c1_perimeter << std::endl;
  36. std::cout << "c1的面积为:" << c1_area << std::endl;
  37. Circle c2(2 * x);
  38. double c2_perimeter = c2.getPerimeter();
  39. double c2_area = c2.getArea();
  40. std::cout << "c2的周长为:" << c2_perimeter << std::endl;
  41. std::cout << "c2的面积为:" << c2_area << std::endl;
  42. Circle c3 = c1;
  43. double c3_perimeter = c3.getPerimeter();
  44. double c3_area = c3.getArea();
  45. std::cout << "c3的周长为:" << c3_perimeter << std::endl;
  46. std::cout << "c3的面积为:" << c3_area << std::endl;
  47. system("pause");
  48. return 0;
  49. }

16.3 运算符重载

要求:编写Integer类使下列代码输出结果为9

  1. int i=2;int j=7;
  2. Integer x(i);
  3. Integer y(j);
  4. cout<<(x+y)<<endl;

代码如下:

  1. #include<iostream>
  2. class Integer{
  3. private:
  4. int value_;
  5. public:
  6. Integer(int v){
  7. value_ = v;
  8. }
  9. Integer& operator + (const Integer& integer){
  10. this->value_ += integer.value_;
  11. return *this;
  12. };
  13. int getValue() const{
  14. return this->value_;
  15. };
  16. };
  17. std::ostream& operator<< (std::ostream& os, const Integer& integer)
  18. {
  19. os << integer.getValue();
  20. return os;
  21. }
  22. int main()
  23. {
  24. int i = 2; int j = 7;
  25. Integer x(i);
  26. Integer y(j);
  27. std::cout << (x + y) << std::endl;
  28. system("pause");
  29. return 0;
  30. }

扩展:算符左++和右++

对于++有两种方式,那么在重载++的时候要怎么区分:

  1. //前置:++i
  2. T& operator++(){
  3. //do something
  4. return *this;
  5. }
  6. //后置:i++
  7. const T operator++(int){
  8. T tmp = *this;
  9. //do something
  10. return tmp;
  11. }

为什么前置++可以返回当前对象的引用而后置++不可以?

因为前置加加是可以允许外部对该对象做修改(它先自增,再返回当前的值)
而后置加加是先返回当前对象的值才自增,故不能返回引用

区别主要在:1、返回值 2、函数参数

代码实现:

  1. #include<iostream>
  2. class Rectangle
  3. {
  4. public:
  5. Rectangle(const int w, const int h) : width_(w), height_(h)
  6. {};
  7. ~Rectangle() {};
  8. // ++i
  9. Rectangle& operator++ ()
  10. {
  11. this->height_ += 1;
  12. this->width_ += 1;
  13. return *this;
  14. }
  15. // i++
  16. //返回类型不能是引用
  17. const Rectangle operator++ (int)
  18. {
  19. //Rectangle temp = *this;
  20. Rectangle temp(*this);
  21. this->height_ += 1;
  22. this->width_ += 1;
  23. return temp;
  24. }
  25. void showVal(){
  26. std::cout << width_ << "---" << height_ << std::endl;
  27. }
  28. private:
  29. int width_;
  30. int height_;
  31. };
  32. int main()
  33. {
  34. Rectangle rct1(40, 10);
  35. Rectangle rct2 = (rct1++);
  36. rct1.showVal();// 输出 41, 11
  37. rct2.showVal();// 输出 40, 10
  38. Rectangle rct3 = (++rct1);
  39. rct1.showVal();// 输出 42, 12
  40. rct2.showVal();// 输出 40, 10
  41. rct3.showVal();// 输出 42, 12
  42. system("pause");
  43. return 0;
  44. }

++i 是自身先自加一,后返回自身

i++ 是先拷贝自身,再自加一,返回自身的拷贝

因此在结果相同的情况下,应优先考虑使用 ++i,例如for循环

17 类的继承和多态

17.1 访问权限控制符

public限定符所修饰的成员变量和函数可以被类的函数、子类函数,以及类对象访问

protected限定符修饰的成员变量和函数可以被类的函数访问,子类函数访问,但是不能被类对象所访问

private限定符修饰的成员变量和函数只能被类函数访问,子类函数无法访问

根据访问权限总结出不同的访问类型,如下所示:






























访问 public protected private
基类 YES YES YES
派生类 YES YES NO
外部类 YES NO NO

继承类型






























继承类型 public protected private
公有继承 public protected private
保护继承 protected protected private
私有继承 private private private
  1. #include <iostream>
  2. class Shape{
  3. public:
  4. void setWidth(int w){
  5. width_ = w;
  6. }
  7. void setHeight(int h){
  8. height_ = h;
  9. }
  10. protected:
  11. int width_;
  12. int height_;
  13. };
  14. // 派生类
  15. class Rectangle : public Shape{
  16. public:
  17. int getArea(){
  18. return (width_ * height_);
  19. }
  20. };
  21. int main(void)
  22. {
  23. Rectangle rect;
  24. rect.setWidth(5);
  25. rect.setHeight(7);
  26. // 输出对象的面积
  27. std::cout << "Total area: " << rect.getArea() << std::endl;
  28. system("pause");
  29. return 0;
  30. }

扩展:在类外访问类中私有成员变量

定义围墙类,使用指针暴力修改类中私有成员的值,实现翻墙代码如下

  1. #include<iostream>
  2. class Wall
  3. {
  4. public:
  5. void setA(int a){
  6. a_ = a;
  7. }
  8. void print(){
  9. std::cout << a_ << std::endl;
  10. std::cout << b_ << std::endl;
  11. }
  12. private:
  13. int a_ = 10;
  14. int b_ = 100;
  15. };
  16. int main()
  17. {
  18. Wall w1;
  19. //w1 为8个字节
  20. std::cout << sizeof(w1) << std::endl;
  21. //把 w1的地址赋值给 Wall指针 w2
  22. Wall* w2 = &w1;
  23. //获取到变量b_的地址
  24. int* w3 = ((int *)w2 + 1);
  25. //修改变量b_的值
  26. *w3 = 0;
  27. w1.print();
  28. system("pause");
  29. return 0;
  30. }

从下图打印结果,可以看出我们已经通过指针暴力修改了围墙类中私有成员变量b_的值
在这里插入图片描述

结论:访问限定符在编译时起作用,当数据映射到内存后,没有任何访问限定符上的区别

17.2 virtual关键字

要求:virtual的用法以及使用场景,代码实现

通过构建虚函数实现多态,使类函数的调用不是在编译期被确定,而是在运行时被确定

  1. class Base
  2. {
  3. public:
  4. virtual void print(){
  5. std::cout << "Base";
  6. }
  7. };
  8. class Derived :public Base
  9. {
  10. public:
  11. void print(){
  12. std::cout << "Derived";
  13. }
  14. };

通过构建纯虚函数,实现抽象类,用于提供模式及后代类应遵循的原则

  1. class Base
  2. {
  3. public:
  4. //类中只要有一个纯虚函数就称为抽象类
  5. //抽象类无法实例化对象
  6. //子类必须重写父类中的纯虚函数,否则也属于抽象类
  7. virtual void func() = 0;
  8. };

17.3 类内存布局及虚表结构

【普通继承】

  1. class Base{
  2. private:
  3. int a;
  4. int b;
  5. public:
  6. void CommonFunction();
  7. //void virtual VirtualFunction();
  8. };
  9. class DerivedClass : public Base{
  10. private:
  11. int c;
  12. public:
  13. void CommonFunction();
  14. //void virtual VirtualFunction();
  15. //void virtual VirtualFunction2();
  16. };

下图是VS2013如上代码编译的结果,从图可知普通类的排布方式,成员变量依据声明的顺序进行排列(类内偏移为0开始),成员函数不占内存空间。而子类的排布方式是先是排布了从父类继承来的的成员变量,接着排布子类的成员变量,成员函数不占字节

在这里插入图片描述

给基类加上虚函数,注释子类DerivedClass,这时的内存排布如下图:内存结构被分成了两个部分,上面是内存分布,下面是虚表。VS编译器把虚表指针放在了内存的开始处(0地址偏移),然后再是成员变量;下面是虚表,在&Base1_meta后面的0表示,这张虚表对应的虚指针在内存中的分布,下面列出了虚函数,左侧的0是这个虚函数的序号,只有一个虚函数,所以只有一项,如果有多个虚函数,会有序号为1,为2的虚函数列出来。编译器是在构造函数创建这个虚表指针以及虚表的

在这里插入图片描述

把子类的注释去掉后,编译发现结果和预期的一样,因为有多个虚函数,所以序号1的虚函数在虚表中被列了出来,它是子类特有的
在这里插入图片描述

【多重继承】

多继承即一个子类可以有多个父类,它继承了多个父类的特性

  1. class Base{
  2. private:
  3. int a;
  4. int b;
  5. public:
  6. void CommonFunction();
  7. void virtual VirtualFunction();
  8. };
  9. class DerivedClass1 : public Base{
  10. private:
  11. int c;
  12. public:
  13. void CommonFunction();
  14. void virtual VirtualFunction();
  15. };
  16. class DerivedClass2 : public Base{
  17. private:
  18. int d;
  19. public:
  20. void CommonFunction();
  21. void virtual VirtualFunction();
  22. };
  23. class DerivedClass : public DerivedClass1, public DerivedClass2{
  24. private:
  25. int e;
  26. public:
  27. void CommonFunction();
  28. void virtual VirtualFunction();
  29. };

其它类的内存分布上面已经讨论过了,这个多重继承的类DerivedDerivedClass内存分布和其它不一样。由上向下,并列排布两个继承的父类DerivedClass1与DerivedClass2,还有自身的成员变量e

在这里插入图片描述

DerivedClass1包含了它的成员变量c,以及Base,Base有一个0地址偏移的虚表指针,然后是成员变量a和b;DerivedClass2的内存排布类似DerivedClass1,DerivedClass2里也有一份Base

【虚继承】

虚继承作用减少过多冗余代码,防止产生二义性

  1. class Base{
  2. private:
  3. int a;
  4. int b;
  5. public:
  6. void CommonFunction();
  7. void virtual VirtualFunction();
  8. };
  9. class DerivedClass : virtual public Base{
  10. private:
  11. int c;
  12. public:
  13. void CommonFunction();
  14. void virtual VirtualFunction();
  15. };

原来是先排虚表指针与Base成员变量,vfptr位于0地址偏移处;但现在有两个虚表指针了,一个是vbptr,另一个是vfptr。vbptr是这个DerivedClass对应的虚表指针,它指向DerivedClass的虚表vbtable,另一个vfptr是虚基类表对应的虚指针,它指向vftable
在这里插入图片描述

因此虚继承的作用是减少了对基类的重复,代价是增加了虚表指针的负担

总结(当基类有虚函数时):

  1. 每个类都有虚指针和虚表;
  2. 如果不是虚继承,那么子类将父类的虚指针继承下来,并指向自身的虚表(发生在对象构造时)
  3. 有多少个虚函数,虚表里面的项就会有多少。多重继承时,可能存在多个的基类虚表与虚指针;
  4. 如果是虚继承,那么子类会有两份虚指针,一份指向自己的虚表,另一份指向虚基表,多重继承时虚基表与虚基表指针有且只有一份

扩展:编译器是如何利用虚表指针与虚表来实现多态的?

当创建一个含有虚函数的父类对象时,编译器在对象构造时将虚表指针指向父类的虚函数,当创建子类的对象时,编译器在构造函数里将虚表指针(子类只有一个虚表指针,它来自父类)指向子类的虚表(虚表里面的虚函数入口地址是子类的)

调用Base *p = new Derived(); 生成的是子类的对象,在构造时,子类对象的虚指针指向子类的虚表,由Derived到Base的转换并没有改变虚表指针,所以p->VirtualFunction,实际上是p->vfptr->VirtualFunction,它在构造的时候就已经指向了子类的VirtualFunction,所以调用的是子类的虚函数,这就是多态

17.4 构造圆柱体类和球类

将圆类作为基类,公有派生出圆柱体类和球类

圆柱体类:
数据成员height,表示高
定义无参构造,半径默认值为1,高度默认值为1
定义有参构造,按指定值创建圆柱高度的设置函数和获取函数
求表面积函数 getArea
求体积函数 getVolume

球类:
定义有参构造,按指定值创建球
求表面积函数 getArea
求体积函数 getVolume

编写测试主函数,创建圆柱体类对象和球类对象,分别输出其表面积和体积

  1. class Cylinder:Circle{
  2. public:
  3. Cylinder(){
  4. radius_ = 1.0;
  5. height_ = 2.0;
  6. }
  7. Cylinder(double r, double h){
  8. radius_ = r;
  9. height_ = h;
  10. }
  11. double getHeight() const{
  12. return height_;
  13. }
  14. void setHeight(const double h){
  15. height_ = h;
  16. }
  17. //求表面积
  18. double getArea() const{
  19. return 2 * M_PI*radius_*radius_ + 2 * M_PI*radius_*height_;
  20. }
  21. //求体积
  22. double getVolume() const{
  23. return M_PI*radius_*radius_*height_;
  24. }
  25. private:
  26. double height_;
  27. double radius_;
  28. };
  29. class Sphere :Circle{
  30. public:
  31. explicit Sphere(double r){
  32. radius = r;
  33. }
  34. //求表面积
  35. double getArea() const{
  36. return 4 * M_PI*radius*radius;
  37. }
  38. //求体积
  39. double getVolume() const{
  40. return 4 * M_PI*radius*radius*radius / 3;
  41. }
  42. private:
  43. double radius;
  44. };
  45. int main() {
  46. Cylinder c1;
  47. Cylinder c2(2.0, 2.0);
  48. double c1_area = c1.getArea();
  49. double c1_volume = c1.getVolume();
  50. std::cout << "c1的表面积为:" << c1_area << std::endl;
  51. std::cout << "c1的体积为:" << c1_volume << std::endl;
  52. double c2_area = c2.getArea();
  53. double c2_volume = c2.getVolume();
  54. std::cout << "c2的表面积为:" << c2_area << std::endl;
  55. std::cout << "c2的体积为:" << c2_volume << std::endl;
  56. Sphere sphere(1.0);
  57. double sphere_area = sphere.getArea();
  58. double sphere_volume = sphere.getVolume();
  59. std::cout << "sphere的表面积为:" << sphere_area << std::endl;
  60. std::cout << "sphere的体积为:" << sphere_volume << std::endl;
  61. system("pause");
  62. return 0;
  63. }

发表评论

表情:
评论列表 (有 0 条评论,264人围观)

还没有评论,来说两句吧...

相关阅读

    相关 编程语言核心

    目录 一、概述 二、 数据类型 三、容器和字符串 四、基础语法 五、流程控制 六、错误处理 七、模块化 八、多线程/并发 九、垃圾回收(GC) 十、编程范式