h5技术网站,wordpress多重标签,我认为的网络营销是什么,西安网站群公司1、多态的概念
1.1 概念
多态的概念#xff1a;通俗来说#xff0c;就是多种形态。具体点就是去完成某个行为#xff0c;当不同的对象去完成时会产生出不同的状态。
2、多态的定义及实现
2.1 多态的构成条件
多态是在不同继承关系的类对象#xff0c;去调用同一函数通俗来说就是多种形态。具体点就是去完成某个行为当不同的对象去完成时会产生出不同的状态。
2、多态的定义及实现
2.1 多态的构成条件
多态是在不同继承关系的类对象去调用同一函数产生了不同的行为。比如 Student 继承了Person。Person 对象买票全价Student 对象买票半价。
继承中构成多态有两个条件
必须通过基类的指针或者引用调用虚函数被调用的函数必须是虚函数且派生类必须对基类的虚函数进行重写 2.2 虚函数
虚函数即被 virtual 修饰的类成员函数称为虚函数。
class Person
{
public:virtual void BuyTicket() { cout 买票-全价 endl;}
};
2.3 虚函数的重写
虚函数的重写覆盖派生类中有一个跟基类完全相同的虚函数即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同称派生类的虚函数重写了基类的虚函数。
// 多态的条件
// 1. 虚函数重写
// 2. 基类的指针或者引用调用虚函数
class Person
{
public:virtual void BuyTicket(){cout 买票-全价 endl;}
};class Student : public Person
{
public:virtual void BuyTicket(){cout 买票-半价 endl;}// 在重写基类虚函数时派生类的虚函数在不加virtual关键字时虽然也可以构成重写// 因为继承后基类的虚函数被继承下来了在派生类依旧保持虚函数属性但是这种写法不是很规范/*void BuyTicket(){cout 买票-半价 endl;}*/
};void Func(Person p)
{p.BuyTicket();
}int main()
{Person ps;Func(ps);Student st;Func(st);return 0;
}
虚函数重写的两个例外
协变基类与派生类虚函数返回值类型不同
派生类重写基类虚函数时与基类虚函数返回值类型不同。即基类虚函数返回基类对象的指针或者引用派生类虚函数返回派生类对象的指针或者引用时称为协变。
class A
{};class B : public A
{};class Person
{
public:virtual A* f() {return new A;}
};class Student : public Person
{
public:virtual B* f() {return new B;}
};
析构函数的重写基类与派生类析构函数的名字不同
如果基类的析构函数为虚函数此时派生类析构函数只要定义无论是否加 virtual 关键字都与基类的析构函数构成重写。虽然函数名不相同看起来违背了重写的规则其实不然这里可以理解为编译器对析构函数的名称做了特殊处理编译后析构函数的名称统一处理成 destructor。
class Person
{
public:virtual ~Person() {cout ~Person() endl;}
};class Student : public Person
{
public:virtual ~Student() { cout ~Student() endl; }
};// 只有派生类Student的析构函数重写了Person的析构函数下面的delete对象调用析构函数才能构成多态才能保证p1和p2指向的对象正确的调用析构函数
int main()
{Person* p1 new Person;Person* p2 new Student;// 析构是虚函数才能正确调用析构函数// 先调用Person::~Person()析构函数然后自动使用operator delete来释放p1指向的内存delete p1;// 先调用Student::~Student()析构函数再调用基类Person::~Person()析构函数。最后自动使用operator delete来释放p2指向的内存delete p2;return 0;
} 注意
普通调用调用函数的对象类型决定调的哪个函数
多态调用调用指针或者引用指向的对象。指向母类调用母类的函数指向女类调用女类的函数
2.4 C11 override和final
C11 提供了 override 和 final 两个关键字可以帮助用户检测是否重写。
final修饰虚函数表示该虚函数不能再被重写
// final 修饰类不能被继承
// final 修饰虚函数不能被重写
class Car
{
public:virtual void Drive() final{}
};class Benz : public Car
{
public:// 报错/*virtual void Drive() {cout Benz-舒适 endl;}*/
};
override检查派生类虚函数是否重写了基类某个虚函数如果没有重写编译报错
class Car
{
public:virtual void Drive(){}
};class Benz :public Car
{
public:virtual void Drive() override {cout Benz-舒适 endl;}
};
2.5 重载、重写覆盖、隐藏重定义的对比 3、抽象类
3.1 概念
在虚函数的后面写上 0则这个函数为纯虚函数。包含纯虚函数的类叫做抽象类也叫接口类抽象类不能实例化出对象派生类继承后也不能实例化出对象只有重写纯虚函数派生类才能实例化出对象。纯虚函数规范了派生类必须重写另外纯虚函数更体现出了接口继承。
class Car
{
public:virtual void Drive() 0;
};class Benz : public Car
{
public:virtual void Drive(){cout Benz-舒适 endl;}
};class BMW : public Car
{
public:virtual void Drive(){cout BMW-操控 endl;}
};void Test()
{Car* pBenz new Benz;pBenz-Drive();Car* pBMW new BMW;pBMW-Drive();
}
3.2 接口继承和实现继承
普通函数的继承是一种实现继承派生类继承了基类函数可以使用函数继承的是函数的实现。
虚函数的继承是一种接口继承派生类继承的是基类虚函数的接口目的是为了重写达成多态继承的是接口。所以如果不实现多态不要把函数定义成虚函数。
4、多态的原理
4.1 虚函数表
// 这里常考一道笔试题sizeof(Base)是多少
class Base
{
public:virtual void Func1(){cout Func1() endl;}
private:int _b 1;
};
通过观察测试我们发现b对象是 8bytes除了_b成员还多一个 __vfptr 放在对象的前面对象中的这个指针我们叫做虚函数表指针v代表 virtualf代表 function。一个含有虚函数的类中都至少都有一个虚函数表指针因为虚函数的地址要被放到虚函数表中虚函数表也简称虚表。 class Base
{
public:virtual void Func1(){cout Base::Func1() endl;}virtual void Func2(){cout Base::Func2() endl;}void Func3(){cout Base::Func3() endl;}
private:int _b 1;
};class Derive : public Base
{
public:virtual void Func1(){cout Derive::Func1() endl;}
private:int _d 2;
};int main()
{Base b;Derive d;return 0;
}
通过观察和测试我们发现了以下几点问题
派生类对象d中也有一个虚表指针d对象由两部分构成一部分是基类继承下来的成员另一部分是自己的成员。基类b对象和派生类d对象虚表是不一样的这里我们发现 Func1 完成了重写所以d的虚表中存的是重写的 Derive::Func1虚函数的重写也叫作覆盖覆盖就是指虚表中虚函数的覆盖。重写是语法的叫法覆盖是原理层的叫法。另外 Func2 继承下来后是虚函数所以放进了虚表Func3 也继承下来了但是不是虚函数所以不会放进虚表。虚函数表本质是一个存虚函数指针的指针数组一般情况这个数组最后面放了一个 nullptr。总结一下派生类的虚表生成
先将基类中的虚表内容拷贝一份到派生类虚表中如果派生类重写了基类中某个虚函数用派生类自己的虚函数覆盖虚表中基类的虚函数派生类自己新增加的虚函数按其在派生类中的声明次序增加到派生类虚表的最后。
虚函数存在哪的虚表存在哪的
虚表存的是虚函数指针不是虚函数虚函数和普通函数一样的都是存在代码段的只是他的指针又存到了虚表中。另外对象中存的不是虚表存的是虚表指针。那么虚表存在哪的呢vs下是存在常量区的。 // 验证虚表在vs下存在代码段
class Base {
public:virtual void func1() { cout Base::func1 endl; }virtual void func2() { cout Base::func2 endl; }
private:int a;
};void func()
{cout void func() endl;
}int main()
{Base b1;Base b2;static int a 0;int b 0;int* p1 new int;const char* p2 hello world;printf(栈:%p\n, b);printf(堆:%p\n, p1);printf(静态区:%p\n, a);printf(常量区:%p\n, p2);printf(虚表:%p\n, *((int*)b1));printf(虚函数地址:%p\n, Base::func1);// 函数名就是函数的地址成员函数比较特殊要加printf(普通函数地址:%p\n, func);return 0;
} 4.2 多态的原理 class Person
{
public:virtual void BuyTicket() { cout 买票-全价 endl; }
};
class Student : public Person
{
public:virtual void BuyTicket() { cout 买票-半价 endl; }
};void Func(Person p)
{p.BuyTicket();
}int main()
{Person Mike;Func(Mike);Student Johnson;Func(Johnson);return 0;
} 观察下图的红色箭头我们看到p是指向 mike 对象时p-BuyTicket 在 mike 的虚表中找到虚函数是 Person::BuyTicket。 观察下图的蓝色箭头我们看到p是指向 johnson 对象时p-BuyTicket 在 johson 的虚表中找到虚函数是 Student::BuyTicket。 这样就实现出了不同对象去完成同一行为时展现出不同的形态。 反过来思考我们要达到多态有两个条件一个是虚函数覆盖一个是对象的指针或引用调用虚函数。 多态调用是运行起来以后到对象的中查找的。普通调用是在编译时确定的。 注意
如果是派生类对象赋值给基类对象不能实现多态为什么因为派生类的成员会拷贝给基类但是不会拷贝虚函数表指针。
4.3 动态绑定与静态绑定
静态绑定又称为前期绑定早绑定在程序编译期间确定了程序的行为也称为静态多态比如函数重载动态绑定又称后期绑定晚绑定是在程序运行期间根据具体拿到的类型确定程序的具体行为调用具体的函数也称为动态多态。
5、单继承和多继承关系的虚函数表
需要注意的是在单继承和多继承关系中下面我们去关注的是派生类对象的虚表模型因为基类的虚表模型前面我们已经看过了没什么需要特别研究的。
5.1 单继承中的虚函数表
class Base
{
public:virtual void func1(){cout Base::func1 endl;}virtual void func2(){cout Base::func2 endl;}
private:int a;
};class Derive : public Base
{
public:virtual void func1() { cout Derive::func1 endl; }virtual void func3() { cout Derive::func3 endl; }virtual void func4() { cout Derive::func4 endl; }
private:int b;
};
观察下图中的监视窗口中我们发现看不见 func3 和 func4。这里是编译器的监视窗口故意隐藏了这两个函数也可以认为是他的一个小bug。那么我们如何查看d的虚表呢下面我们使用代码打印出虚表中的函数。 // 虚函数的地址一定会被放进类的虚函数表
// 打印虚表
typedef void (*VFUNC)();
//void PrintVFT(VFUNC a[])
void PrintVFT(VFUNC* a)
{for (size_t i 0; a[i] ! 0; i){printf([%d]:%p-, i, a[i]);VFUNC f a[i];f(); // 直接用函数指针调用这个函数//(*f)(); // 函数指针解引用调用}printf(\n);
}int main()
{void (*f1)(); // 定义函数指针VFUNC f2; // typedef后可以这样定义Base b;PrintVFT((VFUNC*)(*((int*)b)));Derive d;X x;PrintVFT((VFUNC*)(*((int*)d)));PrintVFT((VFUNC*)(*((int*)x)));return 0;
} 5.2 多继承中的虚函数表
class Base1
{
public:virtual void func1(){cout Base1::func1 endl;}virtual void func2(){cout Base1::func2 endl;}
private:int b1;
};class Base2
{
public:virtual void func1(){cout Base2::func1 endl;}virtual void func2(){cout Base2::func2 endl;}
private:int b2;
};class Derive : public Base1, public Base2
{
public:virtual void func1(){cout Derive::func1 endl;}virtual void func3(){cout Derive::func3 endl;}
private:int d1;
};typedef void (*VFUNC)();
void PrintVFT(VFUNC* a)
{for (size_t i 0; a[i] ! 0; i){printf([%d]:%p-, i, a[i]);VFUNC f a[i];f();}printf(\n);
}
int main()
{Derive d;PrintVFT((VFUNC*)(*(int*)d));//PrintVFT((VFUNC*)(*(int*)((char*)d sizeof(Base1))));Base2* ptr d;PrintVFT((VFUNC*)(*(int*)ptr));return 0;
} 注意
多继承派生类的未重写的虚函数放在第一个继承基类部分的虚函数表中。
6、继承和多态常见的面试问题
6.1 概念查考
1. ( )是面向对象程序设计语言中的一种机制。这种机制实现了方法的定义与具体的对象无关而对方法的调用则可以关联于具体的对象。
A: 继承 B: 模板 C: 对象的自身引用 D: 动态绑定
2. 面向对象设计中的继承和组合下面说法错误的是
A继承允许我们覆盖重写母类的实现细节母类的实现对于女类是可见的是一种静态复用也称为白盒复用
B组合的对象不需要关心各自的实现细节之间的关系是在运行时候才确定的是一种动态复用也称为黑盒复用
C优先使用继承而不是组合是面向对象设计的第二原则
D继承可以使女类能自动继承母类的接口但在设计模式中认为这是一种破坏了母类的封装性的表现
3. 以下关于纯虚函数的说法正确的是( )
A声明纯虚函数的类不能实例化对象
B声明纯虚函数的类是虚基类
C子类必须实现基类的纯虚函数
D纯虚函数必须是空函数
4. 关于虚函数的描述正确的是( )
A派生类的虚函数与基类的虚函数具有不同的参数个数和类型
B内联函数不能是虚函数
C派生类必须重新定义基类的虚函数
D虚函数可以是一个static型的函数
5. 关于虚表说法正确的是
A一个类只能有一张虚表
B基类中有虚函数如果子类中没有重写基类的虚函数此时子类与基类共用同一张虚表
C虚表是在运行期间动态生成的
D一个类的不同对象共享该类的虚表
6. 假设A类中有虚函数B继承自AB重写A中的虚函数也没有定义任何虚函数则
AA类对象的前4个字节存储虚表地址B类对象前4个字节不是虚表地址
BA类对象和B类对象前4个字节存储的都是虚基表的地址
CA类对象和B类对象前4个字节存储的虚表地址相同
DA类和B类虚表中虚函数个数相同但A类和B类使用的不是同一张虚表
7. 下面程序输出结果是什么?
class A
{
public:A(const char* s){cout s endl;}~A(){}
};class B : virtual public A
{
public:B(const char* s1, const char* s2):A(s1){cout s2 endl;}
};class C : virtual public A
{
public:C(const char* s1, const char* s2):A(s1){cout s2 endl;}
};class D : public B, public C
{
public:D(const char* s1, const char* s2, const char* s3, const char* s4):B(s1, s2), C(s1, s3), A(s1){cout s4 endl;}
};
Aclass A class B class C class D Bclass D class B class C class A
Cclass D class C class B class A Dclass A class C class B class D
8. 多继承中指针偏移问题下面说法正确的是( )
class Base1
{
public: int _b1;
};class Base2
{
public:int _b2;
};class Derive : public Base1, public Base2
{
public: int _d;
};int main()
{Derive d;Base1* p1 d;Base2* p2 d;Derive* p3 d;return 0;
}
Ap1 p2 p3 Bp1 p2 p3 Cp1 p3 ! p2 Dp1 ! p2 ! p3
9. 以下程序输出结果是什么
class A
{
public:virtual void func(int val 1){ std::coutA- val std::endl;}virtual void test(){ func();}
};class B : public A
{public:void func(int val 0){ std::coutB- val std::endl; }
};int main(int argc ,char* argv[])
{B* p new B;p-test();return 0;
}
A: A-0 B: B-1 C: A-1 D: B-0 E: 编译出错 F: 以上都不正确 参考答案 1. D 2. C 3. A B.包含纯虚函数的类叫做抽象类也叫接口类 4. B 5. D 6. D 虚基表和虚表不是一种 7. A 构造函数先初始化列表再函数体最后一个打印的是 class D同一继承链中任何虚基类只会在最终派生类中被构造一次即使多个路径上都存在对它的继承。A是B和C的虚基类。构造顺序虚基类总在派生类进行其它任何初始化之前被初始化即使在初始化列表中没有明确调用它派生类构造函数会遵循虚拟基类 - 非虚基类 - 自身成员。当创建D类型的对象时构造过程如下虚基类A的构造函数被调用 - B的构造函数被调用 - C的构造函数被调用 - D的构造函数的主体执行。 8. C 虽然 p1 和 p3 指向相同的位置但是访问的内容不一样 9. B p-test() 是一个普通调用this-func() 是多态调用会调用B类中的 func() 函数。 6.2 问答题 什么是多态 静态多态通过函数重载和运算符重载实现在编译时确定调用哪个函数动态多态1、基类的指针或者引用调用虚函数 2、虚函数完成重写在运行时确定调用哪个函数。指向谁就调用谁的虚函数实现多种形态 什么是重载、重写(覆盖)、重定义(隐藏) 重载同一作用域内函数名相同但参数列表不同参数类型、数量、顺序是静态多态的一种形式。重写派生类重新定义基类中已经存在的虚函数保持函数名、参数列表完全一致。重写的函数在运行时被调用时会调用派生类的实现。即动态多态。重定义派生类定义了与基类同名但不是虚函数的成员函数或变量会隐藏基类的成员。在这种情况下调用基类的成员需要显式地使用作用域解析运算符。 多态的实现原理参考课件 inline 函数可以是虚函数吗 答可以普通调用inline 起作用多态调用inline 不起作用 静态成员可以是虚函数吗 答不能编译报错因为静态成员函数没有 this 指针可以指定类域调用无法构成多态 构造函数可以是虚函数吗 答不能因为对象中的虚函数表指针是在构造函数初始化列表阶段才初始化的。虚函数多态调用要到虚表中找但是虚表指针都还没初始化。 析构函数可以是虚函数吗什么场景下析构函数是虚函数 答可以并且最好把基类的析构函数定义成虚函数。 对象访问普通函数快还是虚函数更快 答首先如果是普通调用是一样快的。如果是多态调用则普通函数快因为构成多态运行时调用虚函数需要到虚函数表中去查找。 虚函数表是在什么阶段生成的存在哪的 答虚函数表是在编译阶段就生成的一般情况下存在常量区。 什么是抽象类抽象类的作用 答参考3、抽象类。抽象类强制重写了虚函数另外抽象类体现出了接口继承关系。