C++ override总结

C++ override是面向对象思想实现的关键。override是指对父类的抽象方法进行重写,以使子类对父类的同一个接口拥有各自不同的实现,即多态。在C++中要实现真正的多态只能是对父类虚方法进行重写,一般情况下还应当加上override关键字,使得编译器帮助我们检查重写是否正确。对于父类的非虚方法不应当进行重写,如果重写了父类的非虚方法,实际上是覆盖(hide),而不可能构成override

eg:

class Base
{
public:
   virtual void say()const
   {
       cout<<"I'm Base"<<endl;
   }

   void run()const
   {
       cout<<"Base run"<<endl;
   }
};

class Derived:public Base
{
public:
    void say()const override
    {
        cout<<"I'm Derived"<<endl;
    }

    void run()const  ///加上overrride后编译器报错
    {
        cout<<"Derived run"<<endl;
    }
};

Derived d;
Base &b = d;
b.say();  ///output:I'm Derived
b.run();  ///output:Base run

C++构成override时,子类中的重写函数声明和父类中的虚函数必须严格一致即函数的参数个数,const属性,返回值类型等都要完全一致,对于返回值类型如果父类返回的是父类的指针或引用类型时,子类可以返回子类类型的指针和引用

eg:

class Base
{
public:
    virtual Base* clone()const
    {
        return new Base(*this);
    }
};

class Derived:public Base
{
public:
    Derived* clone()const override final ///ok,final关键字说明如果Derived有子类,
    {                                    ///则其子类不能对clone函数重写,
        return new Derived(*this);
    }
};

class BaseWrap
{
public:
    virtual Base* clone()const
    {
        return b.clone();
    }
private:
    Base b;
};

class DerivedWrap:public BaseWrap
{
public:
    Derived* clone()const override final ///ok
    {
        return d.clone();
    }
private:
    Derived d;
};

父类的虚函数还可以声明为纯虚函数,拥有纯虚函数的类成为抽象类,抽象类不能创建对象,但可以定义指向抽象类类型的指针和引用,纯虚函数可以实现,但是只能在类外实现,而且即使纯虚函数拥有实现,包含它的类也属于抽象类

class Base
{
public:
    virtual void say()const  = 0;
};

void Base::say()const
{
    cout<<"Base say"<<endl;
}

class Derived:public Base
{
public:
    void say()const override
    {
        Base::say();
        cout<<"Derived say"<<endl;
    }
};

Base b;  ///错误,抽象类不能创建对象
Derived d;
Base &b = d;
b.say();
///output:
///Base say
///Derived say

当c++中的多重继承和函数重写遇上会发生什么呢?

eg:

class Basex
{
public:
    virtual void say()const
    {
        cout<<"Basex say"<<endl;
    }
};

class BaseY
{
public:
    virtual void say()const
    {
        cout<<"Basey say"<<endl;
    }
};

class Derived:public Basex,public BaseY
{
public:
    void say()const override final
    {
        cout<<"Derived say"<<endl;
    }
};

Derived d;
Basex &bx = d;
BaseY &by = d;

bx.say();   ///output:Derived say
by.say();   ///output:Derived say

如果还有虚继承,那么重载又会是怎样的呢?

eg:

class Base
{
public:
    virtual void say()const
    {
        cout<<"Base say"<<endl;
    }
};

class Basex:public virtual Base
{
public:
    void say()const override
    {
        cout<<"Basex say"<<endl;
    }
};

class Basey:public virtual Base
{
public:
    void say()const override
    {
        cout<<"Basey say"<<endl;
    }
};

class Basez:public virtual Base
{
};

///如果虚基类中的虚函数f被其不同的子类重写,而且这些子类又被同一个子类d继承,
///那么d类必须重写虚基类中继承的虚函数f,否则会产生歧义
class Deriveda:public Basex,public Basey   ///错误,虚基类Base中的虚函数say通过两个
{                           ///不同的子类Basex,Basey重写了,因此需要在Derived重写
};

class Derivedb:public Basex,public Basey
{
public:
    void say()const override final
    {
        cout<<"Derivedb say"<<endl;
    }
};

///如果虚基类到其子孙类d的继承路线上,只有其中一个子类dx重写了它的虚函数f
///那么d不必重写虚函数f,而且d中f的实现和dx的实现相同
class Derivedc:public Basex,public Basez
{
};

 Derivedb db;
 Derivedc dc;

 Base &bb = db,&bc = dc;
 Basex &bbx = db,&bcx = dc;
 Basey &bby = db;
 Basez &bcz = dc;

 bb.say();   ///output:Derivedb say
 bbx.say();  ///output:Derivedb say
 bby.say();  ///output:Derivedb say

 bc.say();   ///output:Derivedb say
 bcx.say();  ///output:Derivedb say
 bcz.say();  ///output:Derivedb say
时间: 2024-07-31 07:20:22

C++ override总结的相关文章

关于makefile中变量的多次赋值以及override指令

1 基本原则如下 1.1 原则1 变量的普通赋值是有先后顺序的,后面的赋值会覆盖掉前面的赋值. 1.2 原则2 使用的时候,用的是其前面最后的赋值,就算其后面有使用了override指令的赋值也不会影响这条原则. 1.3 原则3 当使用了override指令定义赋值了变量后,其后对该变量的所有的赋值都是无效的.但是override之前的所有的赋值都是有效的.使用的时候是往前最近原则. 2 override变量.命令行参数和普通变量之间的屏蔽关系 override变量会屏蔽命令行参数,除非用+=:

Scala Study --- override

以前没使用过Scala, 其实我Java也是半截水平\无奈, 学Java的时候刚从C++中挣脱出来,发现Java无比优雅,但很快又对Java种种不信任程序员的设计感到受限. 直到, , 今天遇到了Scala\撒花 Scala的collection设计不能更赞!一段时间后打算专门写篇文章总结Scala,名字就叫"我为什么喜欢Scala!". 废话就不多说了,今天研究了一下Scala的override用法与特点. override --- one of the key words of S

android导入其他工程源码包后出现大量错误提示remove @Override annotation 的解决办法

问题描述: GitHub 下载源码后将其com包内容导入自己android项目中,大量出现错误修改提示 remove @Override annotation 问题解法: 1. 2. 3. 将Compiler compliance level:设置在1.6及以上即可

[转][C++ 11]override and final - write clean and maintainable C++ code

原文: http://arne-mertz.de/2015/12/modern-c-features-override-and-final/ Today I write about a pair of less often discussed, less complicated features introduced in C++11, which are nevertheless useful. Both can provide some additional security and cla

C# 基础 new 、override实现多台区别

一.new只是隐藏父类中的同名方法.基类和父类中都存在这个方法. namespace ConsoleApplication1 { class Program { static void Main(string[] args) { BaseClass bcdc = new DerivedClass(); bcdc.Method2(); //结果:Base - Method2 Console.Read(); } public class BaseClass { public void Method2

override Ext.grid.plugin.RowExpander的方法不起作用

Ext版本4.2 覆盖方法: Ext.override(Ext.grid.plugin.RowExpander, {     setCmp: function (grid) {         var me = this,             rowBodyTpl,             features;         console.log('我是盖子..');         me.callParent(arguments);         me.recordsExpanded 

eclipse @Override报错

三件事需要做: 1 window->Preference->java->Compiler->Compiler compliance lever->设置成最高版本(1.6以上含) 2 project->properties->java compiler->设置成跟上面一样的编译等级. 3 project->properties->Project facets(小方面) -> java设置成跟上面一致的等级. 编译一下应该就没事儿了. ecli

快学Scala 第十九课 (trait的abstract override使用)

trait的abstract override使用: 当我看到abstract override介绍的时候也是一脸懵逼,因为快学scala,只介绍了因为TimestampLogger中调用的super.log依旧是个abstract class,所以必须在方法前加上abstract和override.但是并没有具体介绍如何使用,然后查阅了其他文档,才明白使用方法. 下面的代码定义了超类LoggerEmpty,这个定义意味着该特质只能混入扩展LoggerEmpty的类中. 在特质中声明抽象方法中有

@Override注解

它的作用是对覆盖超类中方法的方法进行标记,如果被标记的方法并没有实际覆盖超类中的方法,则编译器会发出错误警告. /** * 测试Override注解 * @author Administrator */ public class OverrideDemoTest { //@Override public String tostring() { return "测试注解"; } }

java基础随笔-overload和override

今天重温了一下方法重载和方法重写. 首先是方法重写(override)的几点要求: 1.必须继承父类或者实现某接口的方法. 2.方法名称和参数必须和父类(或者实现的接口方法)完全一致. 3.重写的修饰符必须大于等于父类(或者实现的接口方法)的修饰符.比如父类的一个方法protected void haha(); 子类在override这个方法时必须是public void haha() 或者是 protected void haha(). 4.重写不能抛出父类没有定义的非运行时异常. 其次是方法