C#中使用的关键字是base,但用法不完全相同。
1. 调用父类的构造方法
// 父类 public class Father { public Father(String name) { this.name = name; } private String name; } // 子类 public class Son:Father { public Son(String name) : base(name) { } }
注意:
如果没有第14行“:base(name)”,编译器将报告错误:“Father”不包含采用“0”参数的构造函数。因为默认隐式调用
父类不带参数的构造方法,而Father类没有不带参数的构造方法。所以此时需要像第14行那样显示调用父类构造方法。
2. 调用父类的成员
// 父类 public class Father { public String name = "小头爸爸"; public void TellName() { Console.WriteLine("My name is {0}.",name); } } // 子类 public class Son:Father { public new String name = "大头儿子"; public new void TellName() { Console.Write("Father Told: "); base.TellName(); Console.WriteLine("My Father's name is {0}",base.name); Console.WriteLine("My name is {0}.", name); } }
如果不加第13行和第14行的关键字new,编译器将产生2个警告,但不影响程序的运行。因为子类隐藏了父类的成员。
如果第4行和第13行的name成员是private,会少产生一个警告。
3. 子类的子类调用父类中的隐藏成员
一般情况,在调用父类成员时子2代和子1代没什么区别。但如果子1代隐藏父类成员,情况就会不同。
像上面的情况,Son隐藏了父类的成员变量name和成员方法TellName(),如果再有一个类Grandson继承Son,那Grandson调用
Father类中被隐藏的成员时要像这样:
// 子类的子类 public class Grandson : Son { public new String name = "大头孙子"; public new void TellName() { Father f = this as Father; Console.WriteLine("My Grandpa's name is {0}", f.name); Console.WriteLine("My Father's name is {0}", base.name); Console.WriteLine("My name is {0}.", name); } }
也可以使用强制转换,第8行不要,第9行的“f.name”换成“((Father)this).name”。但是不能使用base进行强制转换