【java】this与super使用详解
this三大作用:
this调用属性、调用方法、利用this表示当前对象。
this
this是自身的一个对象,代表对象本身,可以理解为:指向对象本身的一个指针。
this的用法在java中大体可以分为3种:
1.普通的直接引用
这种就不用讲了,this相当于是指向当前对象本身。
2.形参与成员名字重名,用this来区分:
| 
          1
         
          2
         
          3
         
          4
         
          5
         
          6
         
          7
         
          8
         
          9
         
          10
         
          11
         
          12
         
          13
         
          14
         
          15
         
          16
         
          17
         | classPerson {    privateintage = 10;    publicPerson(){    System.out.println("初始化年龄:"+age);}    publicintGetAge(intage){        this.age = age;        returnthis.age;    }}publicclasstest1 {    publicstaticvoidmain(String[] args) {        Person Harry = newPerson();        System.out.println("Harry's age is "+Harry.GetAge(12));    }}        | 
运行结果:
 初始化年龄:10
 Harry's age is 12
可以看到,这里age是GetAge成员方法的形参,this.age是Person类的成员变量。
3.引用构造函数
这个和super放在一起讲,见下面。
super
super可以理解为是指向自己超(父)类对象的一个指针,而这个超类指的是离自己最近的一个父类。
super也有三种用法:
1.普通的直接引用
与this类似,super相当于是指向当前对象的父类,这样就可以用super.xxx来引用父类的成员。
2.子类中的成员变量或方法与父类中的成员变量或方法同名
| 
          1
         
          2
         
          3
         
          4
         
          5
         
          6
         
          7
         
          8
         
          9
         
          10
         
          11
         
          12
         
          13
         
          14
         
          15
         
          16
         
          17
         
          18
         
          19
         
          20
         
          21
         | classCountry {    String name;    voidvalue() {       name = "China";    }} classCity extendsCountry {    String name;    voidvalue() {    name = "Shanghai";    super.value();      //调用父类的方法    System.out.println(name);    System.out.println(super.name);    }     publicstaticvoidmain(String[] args) {       City c=newCity();       c.value();       }} | 
运行结果:
 Shanghai
 China
可以看到,这里既调用了父类的方法,也调用了父类的变量。若不调用父类方法value(),只调用父类变量name的话,则父类name值为默认值null。
3.引用构造函数
| 
           1
          
           2
          
           3
          
           4
          
           5
          
           6
          
           7
          
           8
          
           9
          
           10
          
           11
          
           12
          
           13
          
           14
          
           15
          
           16
          
           17
          
           18
          
           19
          
           20
          
           21
          
           22
          
           23
          
           24
          
           25
          
           26
          
           27
          
           28
          
           29
          
           30
          
           31
          
           32
          
           33
          
           34
          
           35
          
           36
          | classPerson {     publicstaticvoidprt(String s) {        System.out.println(s);     }       Person() {        prt("父类·无参数构造方法: "+"A Person.");     }//构造方法(1)        Person(String name) {        prt("父类·含一个参数的构造方法: "+"A person's name is "+ name);     }//构造方法(2) }    publicclassChinese extendsPerson {     Chinese() {        super(); // 调用父类构造方法(1)        prt("子类·调用父类”无参数构造方法“: "+"A chinese coder.");     }        Chinese(String name) {        super(name);// 调用父类具有相同形参的构造方法(2)        prt("子类·调用父类”含一个参数的构造方法“: "+"his name is "+ name);     }        Chinese(String name, intage) {        this(name);// 调用具有相同形参的构造方法(3)        prt("子类:调用子类具有相同形参的构造方法:his age is "+ age);     }        publicstaticvoidmain(String[] args) {        Chinese cn = newChinese();        cn = newChinese("codersai");        cn = newChinese("codersai", 18);     } } | 
运行结果:
 父类·无参数构造方法: A Person.
 子类·调用父类”无参数构造方法“: A chinese coder.
 父类·含一个参数的构造方法: A person's name is codersai
 子类·调用父类”含一个参数的构造方法“: his name is codersai
 父类·含一个参数的构造方法: A person's name is codersai
 子类·调用父类”含一个参数的构造方法“: his name is codersai
 子类:调用子类具有相同形参的构造方法:his age is 18
从本例可以看到,可以用super和this分别调用父类的构造方法和本类中其他形式的构造方法。
例子中Chinese类第三种构造方法调用的是本类中第二种构造方法,而第二种构造方法是调用父类的,因此也要先调用父类的构造方法,再调用本类中第二种,最后是重写第三种构造方法。
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关键字。
1.this关键字
2.在构造器中调用构造器
 
 
例子:
 
 
关于this()与super()共用的问题:
super()和this ()不能共存,否则编译时会报异常。
Constructor call must be the first statement in a constructor
换句话说就是super()和this ()都必须在构造方法的第一行。
this(有参数/无参数) 用于调用本类相应的构造函数
super(有参数/无参数) 用于调用父类相应的构造函数
而且在构造函数中,调用必须写在构造函数定义的第一行,不能在构造函数的后面使用。
一个构造函数定义中不能同时包括this调用和super调用,如果想同时包括的话,可以在一每构造器中使用一个。
补充:
用类名定义一个变量的时候,定义的应该只是一个引用,外面可以通过这个引用来访问这个类里面的属性和方法,那们类里面是够也应该有一个引用来访问自己的属性和方法纳?JAVA提供了一个很好的东西,就是 this 对象,它可以在类里面来引用这个类的属性和方法。
自我问题:
 http://bbs.csdn.net/topics/391819324?page=1#post-400326846
 
 第一个问题:每次重新调用一个新的构造器之后,this.i的数据都会恢复到原来了0;上一个构造器的this.i的结果没有保存,难道在java中每个构造器都有自己的内存空间地址吗?
 
 
 第二个问题:   System.out.println(tt0.increment().increment().increment());
 //       输出的是test.Test@2b0a141e,这是什么?这是java中的内存地址吗?
 
 
 
 第一个问题:
 构造器(构造函数,Constructor)是一种特殊的函数,它是用来初始化一个“刚刚建立”的对象的方法。不同的构造器重载用于以不同的方式初始化这个对象(其中的逻辑是由编程人员自由控制的)。对于同一个类内的构造器,对于每个对象只能被调用一次,且是由new方法发起调用,可以使用super(调用父类的构造方法)和this(调用本类的构造方法)在不同构造器方法之间跳转,但是每个构造方法最多被调用一遍,同一个类内的构造方法最多有一个是实质上执行的(如果递归,编译器会报错)。this是对当前对象的引用,每次构造器调用时都是作用于一个全新的对象,故每次调用时其类中数据成员的值(那个i是成员变量,在每个对象中有独立的存储空间)一定是未初始化的默认值。
 
 
 static:只有静态成员变量才是整个类共用的。对于非静态成员变量每个对象保存一个独立的值(当然占用的就是对象的空间)。
 
 
 
 第二个问题:
 当没有重写toString方法时,默认会使用Object类提供的toString方法,其规则是:类全名@类对象的hashcode方法返回值。当没有重写hashCode方法时,默认使用Object类的hashCode方法,它是java对一个对象的标识号,可能由不同机制生成,不一定是内存地址,但是可能与之相关(通常实现采用的是将地址转换为int值的方法)。
 
 要搞清楚成员变量和局部变量
 
 Test(int i){ },这个构造器中形参 i 是一个局部变量,作用域是这个构造器的大括号,意思是 声明一个局部变量i,是这个构造器的形参,其值为传入的实参的值,在这个构造器中的i 是指的这个局部变量。
 
 在类中也声明了一个成员变量i,在构造器中也能调用这个变量,但是名字都一样,为了区分应this.i表示这个成员变量,这里的this是指本类的属性
    
Copyright © 2021 半边星
欢迎任何形式的转载,但请务必注明出处。
限于本人水平,如果文章和代码有表述不当之处,还请不吝赐教。
 
                    
                
 
                
            
         浙公网安备 33010602011771号
浙公网安备 33010602011771号