派生类中隐藏基类的虚拟重载函数--从名字空间看待 隐藏

本文通过一个具体的C++代码示例,深入探讨了派生类中成员函数重载与隐藏的区别,以及如何通过使用using声明或重写来解决隐藏问题。

类成员的重载 覆盖和隐藏或许是许多朋友都没彻底明白的语法,在这里,我们试着通过名字空间的解析来理解派生类中隐藏基类的虚拟重载函数.

先来看一个例子:

#include <iostream.h>

class B

{

private:  

int nNumber;

public:   

virtual void test() {       cout << "B::test()/n";  }    

virtual void test(int x) {         nNumber = x; // 将传入的参数赋值给私有成员       cout << "B::test(int x)/n";      }

};

 class D : public B

{

public:

//test(void) 隐藏 B::test(int)   

virtual void test() {       cout << "D::test()/n";    }

};

void main(int argc, char* argv[])

{  D d;          // 派生类的实例 

d.test();       // OK 

d.test(17);     // 产生编译错误的代码行

}

 

有两个概念需要理清:

1.重载

2.名字空间从名字空间的角度看,当C++试图解决某个符号名时,它以由近到远的顺序进行搜索,它首先找本地变量,然后是类成员名,接着是基类名,全局变量……。

一旦编译器解析了名字,它便终止解析过程。如果在不同的名字空间范围碰到同一个符号名,那么它便没有了主意。在你的代码中,一旦编译器确定名字"test"出现在类D中,那么它的名字空间就为类D。不论你调用不带参数的test也好,还是调用带一个整型参数的test(17)也好,编译器都是遵循准这个规则来解决名字问题。对于有整型参数的test(int x)函数,一旦编译器确定函数名test生存在类D中,它便终止范围探查,然后查找与参数匹配的函数。由于它没有在类D中找到带整型参数的函数,所以报错。此时编译器认为函数D::test(void)隐藏了函数B::test(int)。

 

解决方案:

1.重载test(int)class D : public B {public:  virtual void test(int x) { B::test(x); }  ……};

2.引入"using"class D : public B {public: using B::test; // 改写test  virtual void test() {      cout << "D::test()/n"; }};

 

后记:抱歉了,格式很乱.目前还不知道怎么排版~

<think>我们正在讨论C#中的base关键字,特别是用于调用基类构造函数的情况。 根据引用[1][2][3][4],base关键字在派生类构造函数中使用,用于显式调用基类的构造函数。 注意:如果基类有多个构造函数,我们可以通过base传递参数来指定调用哪一个。 另外,如果基类没有无参数构造函数,则派生类必须使用base关键字显式调用基类的某个构造函数,否则会编译错误。 引用[3]提到:当声明一个不带构造函数初始化语句的构造函数时,其实是省略了base()。也就是说,如果我们不写,编译器会默认调用基类的无参构造函数。 但是,如果基类没有无参构造函数,那么我们就必须显式使用base来调用基类的某个有参构造函数。 引用[4]也强调:如果在基类中定义了有参构造函数,那么编译器就不会自动生成无参构造函数。此时,如果派生类没有显式调用基类的构造函数(即没有使用base),并且基类也没有无参构造函数,就会出错。 下面我们通过一个例子来展示如何使用base关键字调用基类构造函数:</think>在C#中,使用`base`关键字调用基类构造函数是面向对象编程中实现继承的重要机制。以下是详细说明和示例: ### 📌 核心用法 1. **语法格式** 在派生类构造函数后使用冒号和`base`关键字,并传递所需参数: ```csharp public DerivedClass(参数列表) : base(基类构造参数) { // 派生类初始化代码 } ``` 2. **调用规则** - **必须作为构造函数第一条语句**(编译器强制)[^2][^4] -基类**没有无参构造函数**时,必须显式调用`base`[^3][^4] - 可指定调用基类的特定重载构造函数(通过参数匹配)[^2][^3] ### 🛠️ 使用场景示例 #### 场景1:基类有带参构造函数 ```csharp public class Animal { protected string Name; public Animal(string name) // 基类带参构造 { Name = name; } } public class Dog : Animal { public int Age; // 必须显式调用基类构造 public Dog(string name, int age) : base(name) // 传递name给基类 { Age = age; // 初始化派生类成员 } } ``` > 💡 若不写`: base(name)`,编译器将报错,因为基类没有无参构造[^4] #### 场景2:选择基类构造函数重载 ```csharp public class Shape { protected int Size; public Shape() => Size = 1; // 无参构造 public Shape(int size) => Size = size; // 带参构造 } public class Circle : Shape { // 显式调用基类的带参构造 public Circle(int customSize) : base(customSize) { } // 调用基类的无参构造 (等价于省略base) public Circle() : base() { } } ``` ### ⚠️ 关键注意事项 1. **执行顺序**:基类构造函数**始终先于**派生类构造函数执行[^4] ```mermaid sequenceDiagram 派生类构造函数->>基类构造函数: 通过base调用 基类构造函数-->>派生类构造函数: 返回执行 派生类构造函数->>派生类成员初始化: 执行剩余代码 ``` 2. **设计建议**: -基类定义了带参构造时,**建议同时显式添加无参构造**,避免派生类调用错误[^3] -基类**只有带参构造**,所有派生类**必须显式使用`base`**[^4] 3. **错误案例**: ```csharp public class Base { public Base(int x) { } } public class Derived : Base { // 错误:基类无默认构造,且未显式调用base public Derived() { } } ``` ### 🌰 实际应用示例 参考矩形面积计算案例[^5]: ```csharp public class Rectangle { protected double length, width; public Rectangle(double l, double w) // 基类带参构造 { length = l; width = w; } } public class Tabletop : Rectangle { public Tabletop(double l, double w) : base(l, w) { // 此处基类构造已先执行 } // 其他方法... } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值