这一节我们来讲一个keyword。就是thiskeyword。

我们还是通过样例来看吧:

class Person
{
private String name;
private int age; Person(String n,int a)
{
name = n;
age = a;
} public void speak()
{
System.out.println(name+":"+age);
}
}
class ThisTest
{
public static void main(String[] args)
{
Person kobe = new Person("KOBE",37);
kobe.speak();
}
}

这个样例我们应该非常熟悉了,前面几节都在用这个样例。我们再来看一看结果:

非常显然,构造函数对对象kobe进行了初始化。

可是我们发现,尽管结果是我们想要的,可是,我们单独看这个构造函数,从可读性的角度我们分析,我们根本就不知道函数传过来的是什么内容,我们能够说一无所知。可阅读性太差了,那么我们再对这个构造函数进行改造:

class Person
{
private String name;
private int age; Person(String name,int age)
{
name = name;
age = age;
} public void speak()
{
System.out.println(name+":"+age);
}
}

唉。这样不是非常清晰了吗。我们一眼就看出这个构造函数要告诉我们初始化对象的姓名和年龄了。我们看结果:

嗯???这是什嘛情况,KOBE人呢?

我们7.4节谈过了构造函数的内存载入过程,可是我们没有提到这里遇到的这样的情况。这样的情况我们能够说是:成员变量和局部变量重名的问题,此时对于栈内存和堆内存中都会变量name和age,而调用的构造函数会自己主动到栈内存中寻找这两个变量。而此时。恰好都有,所以系统就会做一个很有趣的事情。就是把栈内存中的name原赋给本身。而对象的name和age的值事实上在堆内存中,所以结果就是我们刚才看到的结果喽。

那么对于这个问题,我们该怎么解决呢?

java给我们解决方式。那就是用一个keywordthis来区分成员变量和局部变量。我们再来改造:

class Person
{
private String name;
private int age; Person(String name,int age)
{
this.name = name;
this.age = age;
} public void speak()
{
System.out.println(name+":"+age);
}
}

结果:

非常好,KOBE同志又回来了。

所以我们能够说。当成员变量与局部变量重名时,我们能够用this来区分。

那么我们就想明确,this究竟代表什么呢?java语言说,this代表的是对象。

我们还想明确,this代表的是哪个对象?java语言又说,代表的就是当前对象。

专业点的术语是这样定义this的:this就是所在函数所在对象的引用。说简单点就是:this代表本类对象的引用。

我们自己用通俗点的语言来定义:就是哪个对象调用了this所在的哪个函数,this就代表哪个对象,也就是说this就是这个对象的引用。

比方上面的样例中的kobe调用了构造函数Person(String name,int age),那么我们可说this就能够代表kobe这个对象。

那么我们再来对this在内存中的体现过程分析一下,我们继续7.4的过程,仅仅有小的变动。

1.main方法进栈内存。main方法中有一个Person类类型变量kobe;

2.new创建Person对象,在堆内存中创建空间(假如地址为0x0045)。该空间中有两个成员变量name和age;

3.对对象的两个成员变量进行初始化,此时会自己主动选择调用构造函数Person(String n,int a);

4.构造函数Person(String name,int age)进栈内存,參数name="KOBE",age=0也载入入栈。

而此时系统会自己主动为该栈内存中载入一个对象的引用,也就是this,而且把kobe的堆内存地址赋给this;

5.然后在把this.name和this.age的初始化为栈内存中name和age,这样就非常清晰了。this.name和this.age我们能够理解为就this所指堆内存中对象的成员变量,此时对象的初始化完毕;

6.把地址0x0045赋给main方法中的实例变量kobe;

7.构造函数Person(String name,int age)出栈,释放參数name和age和this引用;

8.运行kobe.speak()语句。调用Person类中的speak()方法。则speak方法进栈,此时系统也会为speak方法载入一个this引用,指向堆内存中的对象地址(0x0045);

9.运行打印语句,跳出speak方法,speak方法出栈。释放this引用;

10.跳出main方法,main方法出栈,程序执行结束。

通过上面的过程分析,我们能够简单的总结出这样一个结论:当在函数中须要用到调用函数的对象时。就用thiskeyword。

为了更好的理解这个结论,我们把上面的样例能够标准的写成以下这样:

class Person
{
private String name;
private int age; Person(String name,int age)
{
this.name = name;
this.age = age;
} public void speak()
{
System.out.println(this.name+":"+this.age);
}
}

我们对这个类方法中的全部成员变量都标准的写成了this.成员变量的格式。

那么我们刚開始看的第一个样例为什么没有this,而结果是也是正确的呢?

非常显然,当然对于成员变量与局部变量不重名时。this是能够省略的,但请注意,不是没有,而是省略。

[javase学习笔记]-7.6 thiskeyword的原理的更多相关文章

  1. [javase学习笔记]-7.7 thiskeyword的细节与应用

    这一节我们接着上一节来继续学习thiskeyword. 我们之前在7.5节中的构造函数应注意的细节中提到过一个细节就是构造函数能够调用一般函数,但一般函数不能直接调用构造函数.可是我们没有深究构造函数 ...

  2. javaSE学习笔记(17)---锁

    javaSE学习笔记(17)---锁 Java提供了种类丰富的锁,每种锁因其特性的不同,在适当的场景下能够展现出非常高的效率.本文旨在对锁相关源码(本文中的源码来自JDK 8).使用场景进行举例,为读 ...

  3. JavaSE学习笔记(14)---File类和IO流(字节流和字符流)

    JavaSE学习笔记(14)---File类和IO流(字节流和字符流) File类 概述 java.io.File 类是文件和目录路径名的抽象表示,主要用于文件和目录的创建.查找和删除等操作. 构造方 ...

  4. JavaSE学习笔记(13)---线程池、Lambda表达式

    JavaSE学习笔记(13)---线程池.Lambda表达式 1.等待唤醒机制 线程间通信 概念:多个线程在处理同一个资源,但是处理的动作(线程的任务)却不相同. 比如:线程A用来生成包子的,线程B用 ...

  5. javaSE学习笔记(10)---List、Set

    javaSE学习笔记(10)---List.Set 1.数据存储的数据结构 常见的数据结构 数据存储的常用结构有:栈.队列.数组.链表和红黑树. 1.栈 栈:stack,又称堆栈,它是运算受限的线性表 ...

  6. JavaSE学习笔记(9)---集合类和泛型

    JavaSE学习笔记(9)---集合类和泛型 1.Collection集合 集合概述 在前面我们已经学习过并使用过集合ArrayList<E> ,那么集合到底是什么呢? 集合:集合是jav ...

  7. ZooKeeper学习笔记(二)——内部原理

    zookeeper学习笔记(二)--内部原理 1. zookeeper的节点的类型 总的来说可以分为持久型和短暂型,主要区别如下: 持久:客户端与服务器端断开连接的以后,创建的节点不会被删除: 持久化 ...

  8. javaSE学习笔记(16)---网络编程

    javaSE学习笔记(16)---网络编程 基本概念 如今,计算机已经成为人们学习.工作.生活必不可少的工具.我们利用计算机可以和亲朋好友网上聊天,也可以玩网游.发邮件等等,这些功能实现都离不开计算机 ...

  9. javaSE学习笔记(15) ---缓冲流、转换流、序列化流

    javaSE学习笔记(15) ---缓冲流.转换流.序列化流 缓冲流 昨天复习了基本的一些流,作为IO流的入门,今天我们要见识一些更强大的流.比如能够高效读写的缓冲流,能够转换编码的转换流,能够持久化 ...

随机推荐

  1. Unity图集分割

    using System.IO;using UnityEngine;using UnityEditor; public class TestSaveSprite{ [MenuItem("LL ...

  2. mysql linux查看配置文件my.cnf位置

    原文:mysql linux查看配置文件my.cnf位置 命令: mysql --help | grep 'Default options' -A 1

  3. Java并发和多线程4:使用通用同步工具CountDownLatch实现线程等待

    CountDownLatch,一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待. 用给定的计数 初始化 CountDownLatch.由于调用了 countDown ...

  4. CSS学习(四)

    伪类(Pseudo-classes) CSS伪类是用来添加一些选择器的特殊效果. 伪类的语法: selector:pseudo-class {property:value;} CSS类也可以使用伪类: ...

  5. [Struts2] No result defined for action ... and result input &amp; Invalid field value for field ...

    "No result defined for action ... and result input"错误一般发生在Struts2的拦截器拦截时遇到了问题时.Struts2会将跳转 ...

  6. GIT将本地项目上传到Github(两种简单、方便的方法)

    GIT将本地项目上传到Github(两种简单.方便的方法) 一.第一种方法: 首先你需要一个github账号,所有还没有的话先去注册吧! https://github.com/ 我们使用git需要先安 ...

  7. ThinkPHP5.0框架开发--第10章 TP5.0验证器

    ThinkPHP5.0框架开发--第10章 TP5.0验证器 第10章 TP5.0验证器 ======================================= 今日学习 1.验证器 1) 控 ...

  8. Opencv Mat矩阵中data、size、depth、elemSize、step等属性的理解

    data: uchar类型的指针,指向Mat数据矩阵的首地址.可以理解为标示一个房屋的门牌号: dims: Mat矩阵的维度,若Mat是一个二维矩阵,则dims=2,三维则dims=3,大多数情况下处 ...

  9. correct ways to define variables in python

    http://stackoverflow.com/questions/9056957/correct-way-to-define-class-variables-in-python later say ...

  10. ApacheFlink简介

    对无界数据集的连续处理 在我们详细介绍Flink之前,让我们从更高的层面上回顾处理数据时可能遇到的数据集的类型以及您可以选择处理的执行模型的类型.这两个想法经常被混淆,清楚地区分它们是有用的. 首先, ...