Java开发笔记(四十九)关键字super的用法
前面介绍了如何从bird类继承而来swallow类,按道理子类应当继承父类的所有要素,但是对于构造方法来说,swallow类仅仅继承了bird类的默认构造方法,并未自动继承带参数的构造方法。如果子类想继续使用父类的其它构造方法,就得自己重写心仪的构造方法。例如老鹰属于鸟类,那么可以编写继承自bird类的eagle类,同时要在eagle类内部重新定义拥有多个输入参数的构造方法,由此得到如下所示的eagle类代码:
//定义了一个继承自鸟类的老鹰类
public class eagle extends bird {
// 老鹰类重写了带三个参数的构造方法,则不使用没有输入参数的构造方法
public eagle(string name, int sextype, string voice) {
// 利用super指代父类的构造方法名称
super(name, sextype, voice);
}
}
注意到如上代码用到了关键字super,它的字面意思是“超级的”,但并非说它是超人,而是用super指代父类的名称,所以这里“super(name, sextype, voice)”实际表达的是“bird(name, sextype, voice)”,也就是依然利用了bird类的同名且同参数的构造方法。外部若想创建eagle类的实例,就要调用新定义的带三个参数的构造方法,此时创建实例的代码如下所示:
// 通过构造方法设置属性值
private static void setconstruct() {
// 调用bird类带三个参数的构造方法
bird cuckoo = new bird("杜鹃", 1, "布谷");
system.out.println(cuckoo.tostring());
// eagle类重写了带三个参数的构造方法
eagle eagle = new eagle("鹰" , 0, "啁啁");
system.out.println(eagle.tostring());
}
在类继承的场合,关键字super表示父类,对应的this表示本类。如同this的用法一般,super不但可用于构造方法,还可作为成员属性和成员方法的前缀,例如“super.属性名称”代表父类的属性,“super.方法名称”代表父类的方法。
在中文世界里,性别名称的“雄”和“雌”专用于野生动物,而家畜、家禽的性别应当采用“公”和“母”,比如公鸡、公牛、母鸭、母猪等等。前述的bird类,默认的性别名称为“雄”和“雌”,显然并不适用于家禽。为此几种家禽从bird类派生而来时,需要重新定义它们的性别名称属性,也就是重写setsextype方法,在该方法内部另行对sexname字段赋值。以鸭子类为例,重写方法后的类定义代码如下:
//定义了一个继承自鸟类的鸭子类
public class duck extends bird {
// 定义一个家禽类的性别名称
private string sexname;
public duck(string name, int sex) {
// 利用super指代父类的构造方法名称
super(name, sex, "嘎嘎");
}
public void setsextype(int sextype) {
// 方法内部再调用自身方法,会变成递归调用,如果没有退出机制就变成死循环了
//setsextype(sextype);
// 在方法前面添加前缀“super.”,表示这里调用的是父类的方法
super.setsextype(sextype);
// 修改家禽类的性别名称,此时父类和子类都有同名属性sexname,不加前缀的话默认为子类的属性
sexname = (sextype==0) ? "公" : "母";
//this.sexname = (sextype==0) ? "公" : "母";
}
// 父类的getsexname方法需要重写,否则父类的方法会使用父类的属性
public string getsexname() {
return this.sexname;
}
// 父类的tostring方法需要重写,否则父类的方法会使用父类的属性
public string tostring() {
string desc = string.format("这是一只%s%s,它会%3$s、%3$s地叫。",
this.sexname, getname(), getvoice());
return desc;
}
}
以上的duck类代码,看起来颇有些奇特之处,且待下面细细道来:
1、由于bird类的sexname属性为private类型,表示其为私有属性,不可被子类访问,因此duck类另外定义自己的sexname属性,好让狸猫换太子。
2、重写后的setsextype方法,只有sexname属性才需额外设置,而sextype属性仍遵循父类的处理方式,故此时要调用父类的setsextype方法,即给该方法添加前缀“super.”。
3、因为duck类重新定义了sexname属性,所以与sexname有关的方法都要重写,改为读写当前类的属性,否则父类的方法依旧操作父类的属性。
再来看一个以super修饰成员属性的例子,倘若bird类的sexname属性为public类型,就意味着子类也可访问它,那么duck类便能通过“super.sexname”操作该属性了。此时新定义的duckpublic类代码就变成下面这样:
//演示同名的父类属性、子类属性、输入参数三者的优先级顺序
public class duckpublic extends bird {
public duckpublic(string name, int sex) {
super(name, sex, "嘎嘎");
}
public void setsextype(int sextype) {
super.setsextype(sextype);
// 若想对父类的属性直接赋值,则考虑把父类的属性从private改为public
super.sexname = (sextype==0) ? "公" : "母";
// 父类和子类拥有同名属性,则不带前缀的属性字段默认为子类属性
//sexname = (sextype==0) ? "公" : "母";
//this.sexname = (sextype==0) ? "公" : "母";
}
private string sexname;
public void setsexname(string sexname) {
// 输入参数与类的属性同名,则不带前缀的参数字段默认为输入参数
this.sexname = sexname;
}
}
假设duckpublic类也定义了同名属性,并且另外实现了setsexname方法,于是该类里面将会出现三个sexname,分别是:super.sexname表示父类的属性,this.sexname表示本类的属性,而setsexname内部的sexname表示输入参数。要是三者同时出现两个,必定有一个需要添加“super”或者“this”的前缀,不然编译器哪知同名字段是啥含义?或者说,假如有一个sexname未加任何前缀,那么编译器应该优先认定它是父类属性,还是优先认定它是本类属性,还是优先认定它是输入参数?对于这些可能产生字段名称混淆的场合,java制定了下列的优先级判断规则:
1、方法内部存在同名的输入参数,则该字段名称默认代表输入参数;
2、方法内部不存在同名的输入参数,则该字段名称默认代表本类的成员属性;
3、方法内部不存在同名的输入参数,且本类也未重新定义同名的成员属性,则该字段名称只能代表父类的成员属性;
概括地说,对于同名的字段名称而言,其所表达含义的优先级顺序为:输入参数>本类属性>父类属性。
更多java技术文章参见《java开发笔记(序)章节目录》
上一篇: 设计模式:中介模式
下一篇: Luncene快速入门