JAVA-初步认识-第七章-this关键字的使用场景和原理图解

一. 

this有什么用?我们用例子的形式来体现一下。

下面的截图说明,给人对象一初始化的时候,赋值个姓名。赋值姓名的时候,拿n赋值给name,

DOS显示创建对象时,构造函数初始化成功。

在例子中,我们将“旺财”提供给了n,但是在描述类中,我们要先确定是哪个构造函数中的n。我们看到描述类中,有Person(String n)这个构造函数,至于构造函数里面是怎么写的,我们不知道,对我们而言,里面是封装的。光看函数声明,n到底是什么意思,我们不可能从变量名称上获取n所接受的数据的信息。我们希望提高一下代码的阅读性,写n看不懂,写一个阅读性强一点的,一目了然(那这里的形参可以是字符串么?)。

 直接将构造函数中的形参n,改成字符串name,可以很清楚的知道,我们输入的数据是什么。执行以下源程序,编译没问题,但是运行时出错了,不是我们设想的结果。

解析:分析的同时,回顾一下,各部分在内存中的分布机理。创建的对象位于堆内存中,依据描述事物的类,将成员变量加载入堆内存中的对象空间里,分配地址编号,对成员变量进行默认初始化。在初始化结束前,还要调用构造函数。调用的构造函数是加载入栈内存中,同时将构造函数中的形参name载入栈的构造函数空间内(此时的形参name,会不会有一个默认初始化的值?),由于对象创建时调用了构造函数,给定了输入数据,那么name就有确定数据“旺财”。接着继续执行构造函数里面的语句,里面是将name的值赋值给name。→这里又理解岔了。将构造函数的形参载入栈中的构造函数时,就直接执行构造函数里的语句了,而并不是急于和对象产生关系。形参name载入时,里面的执行语句是将name值赋值给name值,在栈的构造函数里,目前name已经存在了,就不需要向外界搜索name了。这里面name的null值是怎么来的,我觉得是在定义构造函数时,输入形参,就相当于在定义一个name变量,而且没有赋值,那么系统就自动给一个默认的值null。这时构造函数中的name在对自己进行赋值,那么name的值还是默认的null值。对象中的成员变量name并没有改变,这就是局部变量和成员变量重名。→这里,还是理解的不对。堆中对象里的成员变量在,对象分配地址编号后就直接默认初始化了。构造函数是随着创建对象时,成员变量默认初始化后反生的。在这里,我就认为,对象的构造函数初始化发生后,对象的创建才完成。输入的“旺财”到底哪去了?在创建对象时,我们知道对象是调用构造函数进行了初始化。这里的调用就是将 确定的输入数据“旺财”给了构造函数,也就是说,从形参name一加载进栈中的person构造函数空间后,那么name的值就是被赋予了“旺财”。这时,继续执行构造函数里面的内容,name=name,由于构造函数里就有name,自己给自己赋值,相当于没变化。这时,重点来了,栈中的局部变量name和堆中的成员变量name是没有任何关系的,仅仅只是名字相同,而且随着构造函数的执行完毕(构造函数执行结束后,直接弹栈,“旺财”的输入对对象一点意义都没有),系统就认为对象创建结束了。这时再进行p.speak()时,一般函数speak的功能是输出对象中的成员变量name和age,由于对象中两个成员变量的值始终是默认初始化的值,所以仍然是null和0。

 

 

局部变量和成员变量重名,导致构造函数和对象之间无法形成指向。现在我们必须想办法,将两者联系起来。这时,就需要关键字this。this.name就是这个对象的name,将构造函数的值为“旺财”的name赋值给对象的name,将两个name做了区分。

上面这一切发生的缘由是什么,为了提高构造函数的可读性,将形参名和成员变量名统一,提高可读性。但这样导致构造函数无法对对象进行初始化,这时就引入了this关键字,区分成员变量和局部变量。

this就是引用,引用对象。在创建对象调用的构造函数入栈后,为了明确两者之间的对应关系,在明确的输入值赋值给形参name之前,

需要给构造函数中,加入一个this。构造函数中的this指向哪个对象,它就给哪个对象初始化,至于是否真的对对象进行了初始化,并不确定,只是起一个对应指向的作用。其实person被对象调用,进内存时,对象就给了一个this值。通过this关键字,对象标定了构造函数。而且this有明确的值0x0045,不是那种系统默认的一个初始化值。

将person()类中的语句,完整地书写出来,语句如下。

 

执行完person中语句后,将person方法弹栈,在弹栈前,已经根据this.name="旺财",给堆中对象的成员变量name赋值了。

在person方法弹栈结束后,表明对象已经创建完毕。这时,再将对象的地址编号赋值给局部变量p。

紧接着程序为p.speak()。它是用来说出某个对象的姓名和年龄。因此,speak方法在进内存前,也有一个所属关键字this。speak方法的功能是输出name和age,但是栈中的speak方法中,没有定义过name和age局部变量,所以speak访问的全是成员变量,speak能明确哪个对象里的成员变量么?必须要加上this.。

 

目前谈论的添加关键字this的缘由就是,构造函数中的形参和成员变量重名,以及一个this的指引作用。

这时接着执行下面的程序,也就是除了p外,还有p1。

随着p1变量对应的对象新建,person又开始进栈。这里我们在捋一遍这个过程。栈中载在局部变量p1,堆中创建对象。对象的创建是依据描述类中的内容来的,在对象分配完地址编号后,立即在对象中载入成员变量name,age,同时对它们进行默认初始化。在创建对象时,还要接着用构造函数进行初始化。由于创建对象时,输入的数据是“小强”,这对应着person(String name)的构造函数。这样就调用了person(String name)构造函数,为了明确是p1调用的,给了person方法一个this=0x0089的编号,意思是,是0089调用的构造函数。在调用时,将确定的数据“小强”赋值给了person方法中的name局部变量,这时为了使构造函数对对象中的成员进行初始化(局部变量和成员是一样的名字,为了建立联系,必须给出指向),语句书写为this.name=name,也就是说将局部变量name的值“小强”赋值给this为089的name,最后再来一个return,弹栈。这时也标志着0089的对象创建完毕。

最初,person中的局部变量和对象中的成员变量不重名的时候,this不写也是可以的。

但是为了严谨性,也是为了提高可读性,最终程序的书写都标注了this。

 

posted @ 2017-10-13 10:44  前锋营  阅读(998)  评论(0编辑  收藏  举报