C#方法重载(overload)方法重写(override)隐藏(new)
2011-09-15 12:38 javaspring 阅读(270) 评论(0) 编辑 收藏 举报一、重载:同一个作用域内发生(比如一个类里面),定义一系列同名方法,但是方法的参数列表不同。这样才能通过传递不同的参数来决定到底调用哪一个。
值得注意的是,方法重载只有通过参数不同来判断调用哪个方法,而不能通过返回值。至于原因,可以这样理解,你调用一个方法,如果有多个方法同名,系统必须要知道到底你要调用哪一个,参数可以帮助系统在方法入口处得到答案,他根据你给的参数就知道该使用哪个方法。如果只有返回值的不同,很多情况系统是不可能得到有效的判断条件的,比如: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修饰符也将产生警告