重载,重写(覆盖),隐藏

本文解析了C++中的函数调用特性,包括重载、重写与隐藏的概念及区别,并通过实例说明如何避免常见错误。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

JAVA中语言中方法(函数)调用有两种特殊的形态:重载与重写;而C++由于增加了virtual这个虚函数关键字,给函数调用又增加了变数:除了重载、重写(也称覆盖)之外还多了隐藏这么一说。我相信我们绝大多数人写代码时,为了程序的可读性与赶速度的原则,下意识的避开这些类似于脑筋急转弯的语法,尽量使用不同的函数名来区分基类与派生类的函数(当然,不能避开的时候也是非常多的)。可是有时候,有些面试主考官总喜欢拿这三个概念去为难你,考察你的C++基础是否牢固。所以为了面试、这三个概念还是需要我们去区分一下。

  说点个人比较偏激的观点:重载、重写是必须要知道,因为用途太广泛;至于隐藏吗,完全是C++为面试官设计的(^_^等待挨砖)。
 
 1、重载的特征:在同一个类中;函数名字相同;参数不同;virtual 关键字可有可无。

2、重写(覆盖)特征是:分别位于派生类与基类;函数名字相同;参数相同;基类函数必须有virtual 关键字(这点非常要注意)。

  个人认为以上两点比较容易记得住,但是C++的隐藏规则使问题复杂性陡然增加。规则如下:

  1、如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆)。

 2、 如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual关键字。此时,基类的函数被隐藏(注意别与重写混淆)。

说了这么多绕口令,恐怕不容易理解,看个简单的例子吧:
class A{
   public:
   void f(int a);
};
class B: public A{
   public:
   void f(char *p)
};
void main(){
  int i = 1;
  B b;
  A *pa =  &b;
  B *pb = &b;
  pa->f(i);//调用父类A的方法;
  pb->f(i);//这里呢?调用的哪里的方法?
};
按照习惯性思维,觉得main函数中指针pb会调用父类的方法,其实不对,隐藏发生了作用啦,此处符合隐藏的的条件1,所以这里父方法被隐藏掉了,调用的子类的方法,由于参数类型不匹配,编译器直接报错!!

  如果我们在开发中,完全可以通过编译,发现此类错误,及时的修正就无所谓了。但是可怕的是在面试中,对重写、重载、隐藏概念不太熟悉的话,就有可能出错。其实这个例子,是本人几年前的面试时的一道面试题中的一部分(实际的题目还要复杂一些),当时对隐藏这个概念了解得不够,做错了,所以一直记忆犹新。此时拿出来,希望对需要参加面试的板油有些帮助。
本文转自: http://blog.youkuaiyun.com/xgx198831/article/details/7247677
### Java 中方法重载重写以及变量隐藏的概念及用法 #### 方法重载(Method Overloading) 方法重载是指在一个类中可以存在多个具有相同名称但参数列表不同的方法。这些差异可以通过参数的数量、类型或者顺序来体现。方法重载允许开发者在同一作用域下定义功能相似但行为略有不同的方法。 例如,在 `Main` 类中,`p()` 方法通过接受不同类型和数量的参数实现了重载[^1]: ```java class A { public void p(int i) { System.out.println("Integer: " + i); } public void p(double d) { System.out.println("Double: " + d); } } ``` 上述代码展示了两个 `p()` 方法分别处理整型和浮点型数据的情况。 #### 方法重写(Method Overriding) 方法重写指的是子类提供了一个与父类中某个已有方法完全一致的方法签名(包括返回值类型、方法名和参数列表)。在这种情况下,当对象实例化为子类时,会优先调用子类中的版本而非父类中的原始版本。为了实现这一特性,通常需要遵循“声明不变,内容重写”的原则[^2]。 下面的例子说明了如何在继承关系里完成方法覆盖操作: ```java class Parent { public void display() { System.out.println("Parent Method"); } } class Child extends Parent { @Override public void display() { super.display(); // 调用了父类里的display() System.out.println("Child Method"); } } ``` 这里需要注意的是,即使子类已经覆写了某些来自父级的功能模块,仍然能够借助关键字 `super` 来访问那些被替代掉的部分[^5]。 #### 变量隐藏(Variable Hiding) 除了方法之外,Java 还支持字段(属性/成员变量)级别的隐藏机制。这意味着如果子类定义了自己的局部变量,并且它的名字恰好跟它所派生出来的那个超类别下的某个非私有的成员变量相匹配的话,则后者会被前者遮蔽住——也就是说此时再尝试去读取这个共享的名字对应的实体将会指向最近的那个范围内的定义项而不是更远端可能存在的另一个同名条目。 举个例子来说就是这样的情况: ```java class SuperClass { protected int value = 10; } class SubClass extends SuperClass{ private int value = 20; public void showValues(){ System.out.println(this.value); // 输出子类中的value -> 20 System.out.println(super.value); // 输出父类中的value -> 10 } } ``` 在这个片段当中可以看出尽管两者都命名为"value",但由于它们分属不同层次结构之中所以彼此之间互不影响同时也各自独立存在着各自的数值表现形式[^4]. --- ### 总结区别与联系 - **方法重载**关注于同一类内部的不同方法变体创建;主要依据参数配置上的变化来进行区分。 - **方法重写**则涉及跨代际间对于特定动作执行方式更新调整的过程;强调保持接口一致性的同时赋予新的具体逻辑实现能力。 - **变量隐藏**则是关于父子两层架构之下可能出现的数据命名冲突解决方案之一种;虽然表面上看起来像是修改原有项目但实际上只是新增加了一版更加贴近当前上下文环境需求的新诠释而已。 这三个概念共同构成了面向对象编程范式下灵活运用资源的重要组成部分。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值