C# 基类 派生类 方法隐藏 方法重写

本文通过实例对比了C#中方法重写与隐藏的区别,介绍了如何在派生类中正确地重写或隐藏基类的方法,并给出了具体的代码示例。

最近重新学习C#中基础,找到了隐藏基类方法和重写基类方法的一点区别,二者都是在派生类中定义了与基类中相同的方法,相同点派生类对象将执行各自的派生类中的方法,不同点,在向上转型后,重写基类方法的调用的是派生类的方法,而隐藏基类调用的是基类的方法,具体实验代码:

   class Program
    {
        static void Main(string[] args)
        {            
            //fun1被子类覆写了          
            Console.WriteLine("向上转型调用");
            A2 a3 = new A2();
            A2 a2 = new B2();// 向上转型
            a2.fun1();
            a2.fun2();
            Console.WriteLine("基类A2调用");
            a3.fun1();
            a3.fun2();    
            Console.WriteLine("B2子类调用");
            B2 b2 = new B2();
            b2.fun1();
            b2.fun2();
           
        }
        //如果一个方法被子类覆写了 则子类和父类发生转换时 自动调用已经被覆写过的方法
    }

// 基类

    class A2
    {        
        public virtual void fun1()
        {
            Console.WriteLine("A2--->fun1");
        }
        public  void fun2()
        {
            Console.WriteLine("A2--->fun2");
        }
     
    }

  // 派生类

    class B2 : A2
    {
       
        public override void fun1()
        {
            Console.WriteLine("B2--->fun1");
        }
        public  void fun2()
        {
            Console.WriteLine("B2--->fun2");
        }
      
    }

  运行结构:

   向上转型调用
B2--->fun1
A2--->fun2
基类A2调用
A2--->fun1
A2--->fun2
B2子类调用
B2--->fun1
B2--->fun2
请按任意键继续. . .


https://i-blog.csdnimg.cn/blog_migrate/4ff91537b3453b992123d4652ef60870.png

=================================================

重载、重写和隐藏的定义:

重载:同一个作用域内发生(比如一个类里面),定义一系列同名方法,但是方法的参数列表不同。这样才能通过传递不同的参数来决定到底调用哪一个。而返回值类型不同是不能构成重载的。

重写:继承时发生,在子类中重新定义父类中的方法,子类中的方法和父类的方法是一样的 
        例如:基类方法声明为virtual(虚方法),派生类中使用override申明此方法的重写.

隐藏:基类方法不做申明(默认为非虚方法),在派生类中使用new声明此方法的隐藏。

重载时,根据参数选择调用的方法;
重写时,访问父类子类皆调用子类的重写方法;
隐藏时,访问父类则调用父类的方法,子类子类的方法。

隐藏(new)示例:

 
using    System;  
   class    A  
   {  
         public    void    F()    
         {  
               Console.WriteLine("A.F");    
         }  
   }  
   class    B:    A  
   {  
         new    public    void    F()    
         {    
               Console.WriteLine("B.F");    
         }  
   }  
   class    Test  
   {  
         static void Main(string[] args)
         {  
               B    b    =    new    B();  
               b.F();  
               A    a    =    b;    
               a.F();  
         }  
   }  
   输出为  
   B.F  
   A.F  
 
 

重写virtual(虚方法)示例  
   using    System;  
   class    A  
   {  
         public    virtual    void    F()    
         {  
               Console.WriteLine("A.F");    
         }  
   }  
   class    B:    A  
   {  
         public    override    void    F()    
         {    
               Console.WriteLine("B.F");    
         }  
   }  
   class    Test  
   {  
         static    void    Main()    
         {  
               B    b    =    new    B();  
               b.F();  
               A    a    =    b;    
               a.F();  
         }  
   }  
   输出为  
   B.F  
   B.F

补充:重写override一般用于接口实现和继承类的方法改写,要注意

   1、覆盖的方法的标志必须要和被覆盖的方法的标志完全匹配,才能达到覆盖的效果;
   2、覆盖的方法的返回值必须和被覆盖的方法的返回一致;
   3、覆盖的方法所抛出的异常必须和被覆盖方法的所抛出的异常一致,或者是其子类;

http://www.cnblogs.com/whfaillq/archive/2012/02/15/2352722.html


### 基类派生类的关系 在 C# 中,基类(Base Class)和派生类(Derived Class)之间是继承关系。这种关系允许一个类(派生类)从另一个类(基类)中继承属性、方法、字段等成员。基类定义了所有派生类共有的行为和数据结构,而派生类则可以扩展或修改这些行为以适应特定需求。由于继承机制的存在,C# 支持创建复杂的类层次结构[^1]。 #### 继承的特性 - **代码复用**:通过继承,派生类可以直接使用基类中的现有实现,避免重复编写相同的代码。 - **可扩展性**:派生类可以在不改变基类的情况下添加新的功能或者覆盖基类方法来提供不同的实现。 - **多态性**:基于继承,C# 实现了面向对象编程的核心概念之一——多态性,这使得程序能够根据对象的实际类型执行相应的操作。 ### 实现方式 要创建一个派生类,需要使用冒号 `:` 后跟基类名称的形式进行声明。派生类会自动获得基类除构造函数和析构函数外的所有成员[^5]。下面是一个简单的例子说明如何定义基类派生类: ```csharp // 定义基类 public class Animal { public string Name { get; set; } public virtual void MakeSound() { Console.WriteLine("Some sound"); } } // 定义派生类 public class Dog : Animal { // 派生类特有的方法 public void Bark() { Console.WriteLine("Bark!"); } // 覆盖基类方法 public override void MakeSound() { Console.WriteLine("Woof woof"); } } ``` 在这个例子中,`Dog` 类继承自 `Animal` 类,并且重写了 `MakeSound()` 方法以提供特定于狗的声音。此外,它还引入了一个新方法 `Bark()`。 当创建派生类的对象时,首先调用的是基类的构造函数,然后才是派生类自己的构造函数。这样确保了基类部分被正确初始化后再处理派生类特有的部分[^2]。 ### 类型转换 对于基类派生类之间的转换,C# 提供了几种不同的技术: - **隐式转换**:如果存在从派生类到其基类的引用转换,则可以隐式地将派生类实例赋值给基类变量。 - **显式转换**:当需要将基类引用转换为派生类时,必须进行显式的类型转换。 - **安全类型转换**:使用 `as` 运算符尝试转换类型,如果失败则返回 null 而不是抛出异常;或者使用 `is` 关键字检查是否可以成功转换[^3]。 例如,当我们有如下两个类: ```csharp public class BaseClass {} public class DerivedClass : BaseClass {} ``` 我们可以这样进行转换: ```csharp BaseClass baseObj = new DerivedClass(); DerivedClass derivedObj = baseObj as DerivedClass; // 使用 as 进行安全转换 if (derivedObj != null) { // 成功转换后的逻辑 } ``` 以上就是关于 C#基类派生类的基本概念及其关系以及一些常见的实现方式和类型转换技巧。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值