C#方法重载(overload)方法重写(override)隐藏(new)
一、重载:同一个作用域内发生(比如一个类里面),定义一系列同名方法,但是方法的参数列表不同。这样才能通过传递不同的参数来决定到底调用哪一个。
值得注意的是,方法重载只有通过参数不同来判断调用哪个方法,而不能通过返回值。至于原因,可以这样理解,你调用一个方法,如果有多个方法同名,系统必须要知道到底你要调用哪一个,参数可以帮助系统在方法入口处得到答案,他根据你给的参数就知道该使用哪个方法。如果只有返回值的不同,很多情况系统是不可能得到有效的判断条件的,比如:double method();int method();如果你这样调用:method()。系统当然不知道该调用哪个。
示例:
//返回值相同,参数不同
- class BillPayment
- {
- void PayBill(int TelephoneNumber)
- {
- //此方法用于支付固定电话费
- }
- void PayBill(long CustomerNumber)
- {
- //此方法用于支付电费
- }
- void PayBill(long CustomerNumber,double amount)
- {
- //次方法用于支付移动电话费
- }
- }
- }
//返回值相同,参数个数不同
- class Add
- {
- int addtion(int num1,int num2)
- {
- return num1+num2;
- }
- int addtion(int num1,int num2,int num3)
- {
- returnnum1+num2+num3;
- }
- }
//参数不同,返回值不同
- int Method(int[] numbers)
- {
- }
- double Method(double[] numbers)
- {
- }
注:泛型出现后,相同参数,相同返回值的方法也能构成重载,这里不介绍。
二、重写:继承时发生,在子类中重新定义父类中的方法,子类中的方法和父类的方法是一样的,即方法名,参数,返回值都相同。
例如:基类方法声明为virtual(虚方法),派生类中使用override申明此方法的重写.
重写override一般用于接口实现和继承类的方法改写,要注意:
1、覆盖的方法的标志必须要和被覆盖的方法的名字和参数完全匹配,才能达到覆盖的效果;
2、覆盖的方法的返回值必须和被覆盖的方法的返回一致;
3、覆盖的方法所抛出的异常必须和被覆盖方法的所抛出的异常一致,或者是其子类;
4、被覆盖的方法不能为private,否则在其子类中只是新定义了一个方法,并没有对其进行覆盖。
示例:
- namespace 方法重写
- {
- class Program
- {
-
- static voidMain(string[] args)
- {
- BaseClass CO =new ClassOverride();
- CO.SetName("Override");
- }
- //基类
- public classBaseClass
- {
- public virtualvoid SetName(string name)
- {
- Console.WriteLine("基类:我的名字是" + name);
- }
- }
-
- //派生类
- public classClassOverride : BaseClass
- {
- publicoverride void SetName(string name)
- {
- Console.WriteLine("Override:我的名字是" + name);
- }
- }
-
- }
- }
在方法重写中,我们介绍了虚方法,那么虚方法和抽象方法有什么区别呢?
1.虚方法有一个实现部分可以被子类继承,从而使子类获得和基类相同的方法,另外也为派生类提供了覆盖该方法的选项。相反,抽象方法没有提供实现部分,是一种强制派生类覆盖的方法(否则派生类不能成具体类)
2.(abstract)抽象方法只能在抽象类中声明,(virtual)虚方法不是。
3.(abstract)抽象方法必须在派生类中重写而(virtual)虚方法不必。
4.(abstract)抽象方法不能声明方法实体,虚方法可以。
个人感觉,方法重载和方法重写都有点像多态,同一个方法的多种形态:执行“相同”的方法,却通过它们自己的实现代码来实现,即同一种方法,不同的对象会产生不同的结果。
三、隐藏(方法):基类方法不做申明(默认为非虚方法),在派生类中使用new声明此方法的隐藏。隐藏时,访问父类则调用父类的方法,访问子类则调用子类的方法。
示例:
- namespace 隐藏
- {
- class Program
- {
-
- static voidMain(string[] args)
- {
- ClassNew CN =new ClassNew();
- CN.SetName("new");
-
- BaseClass BC =CN;
- BC.SetName("基类");
- }
- //基类
- public classBaseClass
- {
- public voidSetName(string name)
- {
- Console.WriteLine("基类:我的名字是" + name);
- }
- }
- //派生类
- public classClassNew : BaseClass
- {
- //这里如果不使用new,将生成警告!
- new publicvoid SetName(string name)
- {
- Console.WriteLine("new:我的名字是" + name);
- }
- }
- }
- }
运行结果:new:我的名字是new
基类:我的名字是基类
如果上述扔不好理解,我们再举一个隐藏静态变量的例子
- namespace 隐藏继承成员
- {
- class Program
- {
- public classBaseClass
- {
- public static int A = 123;
- }
-
- public classClassNew : BaseClass
- {
- new public static int A = 456;
-
- static voidMain(string[] args)
- {
- Console.WriteLine(A);
- }
-
- }
- }
- }
运行结果:456
注:不仅有方法隐藏,还有其他的,通过继承隐藏名称一般采用下列形式之一:
1.引入类或结构中的常数、指定、属性或类型隐藏具有相同名称的所有基类成员。
2.引入类或结构中的方法隐藏基类中具有相同名称的属性、字段和类型。同时也隐藏具有相同签名的所有基类方法。
3.引入类或结构中的索引器将隐藏具有相同名称的所有基类索引器。
4.在同一成员上同时使用 new 和 override 是错误的。
另外,在不隐藏继承成员的声明中使用 new 修饰符将生成警告。在隐藏继承成员的生命中不使用new修饰符也将产生警告