Loading

JAVA_SE基础——40.super关键字

只要this关键字掌握了,super关键字不在话下,因为他们原理都差不多的。。


this&super

什么是this,this是自身的一个对象,代表对象本身,可以理解为:指向对象本身的一个指针。当你想要引用当前对象的某种东西,比如当前对象的某个方法,或当前对象的某个成员,你便可以利用this来实现这个目的。要注意的是this只能在类中的非静态方法中使用,静态方法和静态的代码块中绝对不能出现this。his也可作为构造函数来使用。在后面可以看到

而什么是super,可以理解为是指向自己超(父)类对象的一个指针,而这个超类指的是离自己最近的一个父类。super的作用同样是可以作为构造函数使用,或者是获取被局部变量屏蔽掉的父类对象的某个同名变量的值。


当子类重写父类的方法后,子类对象将无法访问父类被重写的方法,为了解决这问题,在Java中专门提供了一个super关键字用于访问父类的成员。例如访问父类的成员变量、成员方法和构造方法。


①使用super关键字调用父类的成员(变量和方法)

格式:

super.成员变量

super.成员方法([参数1,参数2……])


测试代码1(使用super调用父类的成员):

package day08;
//定义Animal类
class Animal_3{
	
	String name = "动物"; 
	//定义动物叫的方法
	void shout(){
		System.out.println("动物发出声音中……");
	}
	
}
//定义Cat类继承动物类
class Cat_3 extends Animal_3{
	
	String name = "猫科" ;
	//重写父类的shout()方法
	void shout(){
		super.shout();
	}
	//定义打印name的方法
	void printName(){
		System.out.println("name = "+super.name);
	}
}

public class Demo4 {
	public static void main(String[] args){
		Cat_3 cat = new Cat_3();
		cat.shout();
		cat.printName();
	}
}
运行结果:

动物发出声音中……
name = 动物


代码测试1解释:定义了一个Cat类继承Animal_3类,并重写了Animal_3类的shout()方法。在子类Cat的shout()方法中使用super.shout()  调用了父类被重写的方法,在printName()方法中使用了 super.name 访问了父类的成员变量。  所以从结果看出而得出的结论是:子类通过super关键字可以成功的访问父类成员变量和成员方法。


②使用super关键字调用父类的构造方法

格式:

super([参数1,参数2……])


测试代码2(使用super调用父类的构造方法):

<span style="font-size:18px;">package day08;
class Animal_4{
	//定义Animal类有参的构造方法
	public Animal_4(String name){
		System.out.println("有参构造方法被调用中……");
		System.out.println("我是一只"+name);
	}
}
//定义Dog_2类继承Animal类
class Dog_2 extends Animal_4{
	public Dog_2(){
		super("哈士奇");
	}
}
public class Demo5 {
	public static void main(String[] args){
		Dog_2 dog = new Dog_2();
	}
}</span>

运行结果:

有参构造方法被调用中……
我是一只哈士奇


代码测试2解释:在实例化Dog_2对象时候一定会调用Dog_2类的构造方法。从运行结果可以看出,Dog_2类的构造方法被调用时父类的构造方法也被调用。

需要注意的是:通过super调用父类构造方法的代码必须位于子类构造方法的第一行,并且只能出现一次。


P.S. 

如果把super(“哈士奇”); 这行代码删除    编译器会报错,出错原因:实际参数列表和形式参数列表长度不同,在子类的构造方法中一定会调用父类的某个构造方法。这时子类构造方法会通过super指定调用父类的哪个构造方法,如果没有指定,在实例化子类对象时候,会自动调用父类无参的构造方法。  为了解决测试代码2的问题,可以在子类中显示地调用父类中已有的构造方法,当然也可以在父类定义无参构造方法。

解决问题后:

package day08;

class A{
	int x = 10 ;
	public A(){
		System.out.println("A类无参构造方法被调用");
	}
	public A(int x){
		System.out.println("A类的有参构造方法被调用");
	}
}
class B extends A{
	//显式构造无参代码
	public B(){
		//隐式super()
		//方法体无代码
	}
}
public class Demo6 {
	public static void main(String[] args){
		B b = new B();
		
	}
}
运行结果:

A类无参构造方法被调用


从结果看出,子类在实例化时默认调用了父类的无参的构造方法。

从这代码得出结论: 在定义一个类的时候,如果没有特殊需求,尽量在类中定义一个无参的构造方法,避免被继承时出现错误。



测试代码3(为何实例对象B,A对象的无参构造方法也被调用)

package day08;

class A{
	public A(){
		System.out.println("A类无参构造方法被调用");
	}
}
class B extends A{
	public B(){
		System.out.println("B类无参构造方法被调用");
	}
}
public class Demo6 {
	public static void main(String[] args){
		B b = new B();
		
	}
}
运行结果:

A类无参构造方法被调用
B类无参构造方法被调用


大家思考下,为何A类 的无参构造方法也被调用了 ,按道理来说 不是实例B类 然后只打印“B类无参构造方法被调用”


测试代码3解释:B类的构造方法被调用时父类B类的构造方法也被调用了。B类的构造方法第一行有一个隐私的super();这点和 this关键字是一样的理解的所以才可以调用父类的无参构造方法。


super关键字与this关键字的区别:
1. 代表的事物不一致。
1. super关键字代表的是父类空间的引用。
2. this关键字代表的是所属函数的调用者对象。
2. 使用前提不一致。
1. super关键字必须要有继承关系才能使用。
2. this关键字不需要存在继承关系也可使用。
3. 调用构造函数的区别:
1. super关键字是调用父类的构造函数。
2. this关键字是调用本类的构造函数。


super关键字调用父类构造方法要注意的事项:
1. 如果在子类的构造方法上没有指定调用父类的构造方法,那么java编译器会在子类的构造方法上面加上super()语句。
2. super关键字调用父类的构造函数时,该语句必须要是子类构造函数中的第一个语句。
3. super与this关键字不能同时出现在同一个构造函数中调用其他的构造函数。因为两个语句都需要第一个语句。


super(参数):调用父类中的某一个构造函数(应该为构造函数中的第一条语句)。
this(参数):调用本类中另一种形式的构造函数(应该为构造函数中的第一条语句)。


总结调用变量和方法:


用于变量时:this用于访问实例变量,以区分局部局部变量,super用于访问父类的属性(前提是访问控制权限允许)
用于方法时:隐含调用本类的方法,super用于访问父类的方法(前提是访问控制权限允许)



交流学习企鹅:654249738



posted @ 2015-10-09 22:46  脚本叔叔  阅读(187)  评论(0编辑  收藏  举报