C# 类型运算符重载在类继承中的调用测试

这是一篇晦涩难懂的片面的研究

一,简单的继承层次

    class CA {
    }
    class CB : CA{
    }
    class CC : CB{
        }
    }
    void Test(CA oa){//CATest
        Debug.Log ("CA===============");
    }
    void Test(CB oa){//CBTest
        Debug.Log ("CB===============");
    }
    void Test(CC oa){//CCTest
        Debug.Log ("CC===============");
    }

//测试代码如下:
    CC oc = new CC ();
    Test (oc);    

在这种情形下调用Test(oc)有如下规律:<通过注释掉其它函数进行测试>

  1. 若CATest , CBTest, CCTest三个重载函数都存在,则Test(oc)将调用CCTest
  2. 若只有CATest, CBTest二个重载函数,则Test(oc)将调用CBTest
  3. 若三个函数只有一个存在,则Test(oc)即调用该函数。

由此我们得知,Test(oc)调用时,编译器会由oc的继承层次由子到父的优先级去匹配重载函数的形参。这也符合正常逻辑。

二,类中有运算符重载的继承

   class CA {
    }
    class CB : CA{
    }
    class CC : CB{
/*        MSDN类型转换的要求
        1.操作数必须是封闭类型
        2.类A到类B的类型转换定义不能在类C中进行(即2个类的转换不能在第3个类中定义, 如下面的参数定义)*/
        public static implicit operator bool(CC ot/*不能是CA ot或 CB ot */){
            Debug.Log ("bool================");
            return ot != null;
        }
    }
    void Test(CA oa){//CATest
        Debug.Log ("CA===============");
    }
    void Test(CB oa){//CBTest
        Debug.Log ("CB===============");
    }
    void Test(CC oa){//CCTest
        Debug.Log ("CC===============");
    }
   void Test(bool b){//boolTest     Debug.Log ("b===============")   }
//测试代码如下:
    CC oc = new CC ();
    Test (oc);    

此情形下boolTest重载函数和CATest, CBTest, CCTest的任何一个重载都冲突,原因如下:

当调用Test(oc)时,编译系统将oc与Test的四个重载函数的参数进行匹配,却发现四个都能匹配成功。Test(bool b)通过CC类的bool类型符重载而匹配。

CATest, CBTest, CCTest三个重载函数由于形参CC,CB,CA是继承关系,在进行匹配时是有优先级的,由于ot是CC类型的,所以优先级CC>CB>CA,因此这个三个重载函数间没有冲突,编译器明确知道该调用哪个重载。而bool重载与CC,CB,CA在类型转换时是同优先级,因此编译系统不知道该调用bool重载还是CC,CB,CA的三个重载了。

若将bool重载由类CC移到类CA中,其它代码不变,测试代码不变。经测试,boolTest,CATest, CBTest, CCTest四个重载可以共存,即boolTest与其它任何一个重载都不冲突。

  1. boolTest,CATest, CBTest, CCTest同时存在,Test(oc)调用了CCTest
  2. boolTest,CATest, CBTest同时存在,Test(oc)调用了CBTest
  3. boolTest, CATest同时存在,Test(oc)调用了CATest
  4. boolTest,CATest, CBTest, CCTest只有一个存在,则调用此存在
  5. 只有boolTest存在时,该重载函数也被调用

这说明了基类的类型重载运算符的调用优先级低于父子层级转换的优先级,如情形5,在只有bool重载运算符时才会被调用。

本类的类型重载运算符的优先级等于父子层次转换的优先级。

时间: 2024-12-25 21:22:48

C# 类型运算符重载在类继承中的调用测试的相关文章

第九周(运算符重载时间类)

/* *copyright(c) 2015,烟台大学计算机学院 *All rights reserved. *文件名称:第九周(运算符重载时间类) *作者:王忠 *完成日期:2015.5.13 *版本号:v1.0 * *问题描述:实现Time类中的运算符重载.定义对时间对象的自增和自减一目运算符 //一目运算符的重载 CTime operator++(int);//后置++,下一秒 CTime operator++();//前置++,下一秒,前置与后置返回值不一样 CTime operator-

(C++)C++类继承中的构造函数和析构函数

思想: 在C++的类继承中, 建立对象时,首先调用基类的构造函数,然后在调用下一个派生类的构造函数,依次类推: 析构对象时,其顺序正好与构造相反: 例子: #include <iostream> using namespace std; class Shape{ public: void Draw() {cout<<"Base::Draw()"<<endl;} void Erase() {cout<<"Base::Erase()

第九周(运算符重载分数类)

/* *copyright(c) 2015,烟台大学计算机学院 *All rights reserved. *文件名称:第九周(运算符重载分数类) *作者:王忠 *完成日期:2015.5.13*版本号:v1.0 * *问题描述:定义分数的一目运算+和-,分别代表分数取正和求反,将"按位取反运算符"~重载为分数的求倒数运算. *输入描述: *程序输出: #include <iostream> #include <Cmath> using namespace std

第八周-运算符重载-Time类中的运算符重载

代码如下: #include <iostream> using namespace std; class CTime { private: unsigned short int hour; //时 unsigned short int minute; //分 unsigned short int second; //秒 public: CTime(int h=0,int m=0,int s=0); void setTime(int h,int m,int s); void display();

C++类继承中内存的布局

1 前言       了解你所使用的编程语言究竟是如何实现的,对于C++程序员可能特别有意义.首先,它可以去除我们对于所使用语言的神秘感,使我们不至于对于编译器干的活感到完全不可思议:尤其重要的是,它使我们在Debug和使用语言高级特性的时候,有更多的把握.当需要提高代码效率的时候,这些知识也能够很好地帮助我们. 本文着重回答这样一些问题: 1* 类如何布局? 2* 成员变量如何访问? 3* 成员函数如何访问? 4* 所谓的"调整块"(adjuster thunk)是怎么回事? 5*

第九周(运算符重载复数类)

/* *copyright(c) 2015,烟台大学计算机学院 *All rights reserved. *文件名称:第九周(运算符重载) *作者:王忠 *完成日期:2015.5.13 *版本号:v1.0 * *问题描述:定义Complex类中的<<和>>运算符的重载,实现输入和输出, *输入描述: *程序输出: #include <iostream> using namespace std; class Complex { public: Complex(){rea

关于 python 类与继承中方法调用 的 一个小知识点记录

(1)D类,多继承于C类与B类,C类与B类继承于A类.C类中没用__init__(), C类的实例化会先寻找第一个继承类是否存在__init__(),也就是B类的__init__().因为python3中使用的是广度优先的方法,寻找路径为D-->B-->C-->A 关于其他方法的继承,也是这个顺序. class A(object): def __init__(self): print("i am A") def call(self): print("A CA

虚基类 继承中的构造函数

/* *Copyright (c)2014,烟台大学计算机与控制工程学院 *All rights reserved. *文件名称:d.cpp *作 者:张旺华 *完成日期:2015年6月1日 *版 本 号:v1.0 */ #include <iostream> using namespace std; class Base { public: Base(char i) { cout<<"Base constructor. --"<<i<<

关于友元函数,运算符重载和类的类型转换

一 关于运算符的重载:运算符中有10种是不可重载的:sizeof static_cast<type>. dynamic_cast<type>.* const_cast<type>:: typeid?= reinterpret_cast<type>有4种是只可以在类中重载的:() [] = -> (1)同时在重载时需要有一个参数是用户自己定义的类成员,否则无法使用 (2)运算符的使用方式,及所需参数个数与重载前所需的参数个数相同 重载的格式: retur