Java中的this和super实例浅析
要说this和super就不得不说Java的封装和继承了,首先说封装,这是一种思想,算不上一种技术,核心思想就是将对象的同一行为和状态看成是一个整体,将无需对外界暴露的属性和方法隐藏起来,比如一些方法的具体实现和一些私有的变量,通过公共的方法提供对属性的操作,从而提高安全性。
class Person { private String name = "无名氏";//私有的内部成员变量 private int age = 20;//私有的内部成员变量 public void sayHello(){//公有的方法,外界可以通过调用这个方法获得需要的功能 System.out.println("我的名字是"+name+",今年"+age+"了"); } public String getName() {//公有的方法,外界通过操作这个方法获取私有变量值 return name; } public void setName(String name) {//公有方法,外界通过操作这个方法设置公有变量的值 this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } }
然而这种封装也不是绝对的,当我们需要一些很相似的类的时候,往往希望能够通过一个类派生出许多的拥有共同的特性的类,比如通过动物这个类,派生出猫狗鸡鸭这些类,他们都拥有动物的特性,但是又各有特点。这就是Java中的继承,通过继承,我们可以获得父类所暴露给子类的所有成员,而又可以在父类的基础上构建自己特有的属性。
public class StaticDemo { public static void main(String[] args) { Student s = new Student(); s.age = 20;//从父类继承的属性 s.name = "张同学"; s.stuNumber = 10;//子类特有的成员变量 s.sayHello();//从父类继承的方法 s.study(); } } class Person { protected String name = "无名氏";// 子类共享的成员变量 protected int age = 20;// 子类共享的成员变量 public void sayHello() {// 公有的方法,外界可以通过调用这个方法获得需要的功能 System.out.println("我的名字是" + name + ",今年" + age + "了"); } } class Student extends Person { public int stuNumber = 20;// 学生特有的学号 public void study() {// 学生特有的方法,学习 System.out.println("好好学习!"); } }
至此,我们已经大概了解封装和继承,那下面就来看看this和super关键字是什么意思。
super关键字代表的是父类对象,this关键字代表的是当前对象。要理解这些,首先我们要看看子类在创建对象的时候执行了什么。为了方便起见,在此都使用无参构造方法。
编译器找到子类无参构造方法。
执行子类无参构造方法。
在子类无参构造方法的方法体中,第一句代码是隐藏得super();这句代码的意思是执行父类构造方法。
跳转到父类构造方法中,执行父类构造方法,注意,此时子类构造方法并未执行完毕,并且若父类还有父类,则一直执行上一层父类的构造方法,直到找到所有类的祖宗Object类。
将父类的对象的引用存入子类的super中,即子类中的super代表的是父类实体。然而在父类中,super代表的又是父类的父类的实体。
将创建出的当前类的引用存入this。
public class StaticDemo { public static void main(String[] args) { GirlStudent g = new GirlStudent(); g.sayName(); } } class Person { protected String name = "无名氏";// 子类共享的成员变量 protected int age = 20;// 子类共享的成员变量 public void sayHello() {// 公有的方法,外界可以通过调用这个方法获得需要的功能 System.out.println("我的名字是" + name + ",今年" + age + "了"); } } class Student extends Person { public int stuNumber = 20;// 学生特有的学号 public String name = "张同学"; public String getSuperName() {//获取当前类的父类名字的方法,对于student是父类,对于girlstudent则是爷爷类 return super.name;//父类中的super代表的是父类的父类,即student的父类即person类,可以看出子类一旦初始化,将会创建出所有父类对象,一直到object } } class GirlStudent extends Student{ public String name = "小红"; public void sayName(){ System.out.println("自己的名字是:"+this.name+",父类的名字是:"+super.name+",爷爷类的名字是:"+super.getSuperName()); } }
由此可以看出super的用途:用于调用子类隐藏的父类的属性,比如上述代码中的name属性,还有一个比较重要的用途就是用于调用被子类覆写的方法,此时方法名相同,必须使用super才能调用父类的方法。
this的用途:在变量定义不明确时,用来确定调用的具体是哪个变量。
public void setName(String name){ this.name=name; }
在上面的代码中由于传入的变量也是name,类中本来就存在一个变量也叫name,此时将不明确name是哪个,所以使用this进行区分。
PS:super和this的异同:
super(参数):调用基类中的某一个构造函数(应该为构造函数中的第一条语句)
this(参数):调用本类中另一种形成的构造函数(应该为构造函数中的第一条语句)
super: 它引用当前对象的直接父类中的成员(用来访问直接父类中被隐藏的父类中成员数据或函数,基类与派生类中有相同成员定义时如:super.变量名 super.成员函数据名(实参)
this:它代表当前对象名(在程序中易产生二义性之处,应使用this来指明当前对象;如果函数的形参与类中的成员数据同名,这时需用this来指明成员变量名)
调用super()必须写在子类构造方法的第一行,否则编译不通过。每个子类构造方法的第一条语句,都是隐含地调用super(),如果父类没有这种形式的构造函数,那么在编译的时候就会报错。
super()和this()类似,区别是,super()从子类中调用父类的构造方法,this()在同一类内调用其它方法。
super()和this()均需放在构造方法内第一行。
尽管可以用this调用一个构造器,但却不能调用两个。
this和super不能同时出现在一个构造函数里面,因为this必然会调用其它的构造函数,其它的构造函数必然也会有super语句的存在,所以在同一个构造函数里面有相同的语句,就失去了语句的意义,编译器也不会通过。
this()和super()都指的是对象,所以,均不可以在static环境中使用。包括:static变量,static方法,static语句块。
从本质上讲,this是一个指向本对象的指针, 然而super是一个Java关键字。