this和super是Java的两个关键字。

先明确一个问题,有人错误的认为它们是对象里的“属性”,这只能怪老师没有讲清楚计算机的本质了。因为计算机的处理器只能用指令去处理数据,像C语言之类的容易理解,就是一个个的方法调用,对数据进行处理。那面向对象语言,确实是用对象调用方法啊,怎么回事?

好办,编译器耍个花样,将对象当做方法的参数就是了。比如

class Test{

public void test(){}

}

Test t = new Test();

t.test();

计算机怎么处理呢?

假设我们把以上代码映射到面向过程的语言,大体是这个样子的:

void Test::test(final Test this){
      }

当使用t.test();时,计算机实际处理为:

Test::test(t);//看到了吗?在方法中,this相当于形式参数,而调用方法的对象是实际参数。

也就是说this是实例方法中的第一个参数。对于static的方法,因为没有这个参数,所以就不能使用this了。

this和super的基本功能这里不想探讨,只想说明一下它们在本质上的差别。

有提法分别称之为this引用和super引用,为了表达的方便,这么说也无可厚非。但是从本质上来说,super并非一个引用,仅仅是一个起到指示作用的关键字而已,这与this在本质上是不同的。很简单的两行代码就说明问题了:

class Main {
    public void test() {
          System.out.println(this);
          System.out.println(super);
   }
}   

编译报错,super后面缺少.。也就说,至少在Java语言看来,super并不是一个合法的引用,而显然this是可以的。

为了表达的方便,我们仍然称super为"super引用"。这种意义下,super引用和this引用有什么差别吗?差别当然有,最重要的是,类型不 同,super的类型为父类类型引用,而this为当前类类型的引用。相同点呢?那就是这两个都是指向“当前对象”。不是说“父类对象”吗?啥,哪来的“ 父类对象”这个概念啊?

言多必失,还是看代码吧。

class Base {
      int i = 5;
      public void test() {
           System.out.println("In Base:" + this.i);
      }
}  

public class Test extends Base {
       int i = 55;
       public void test() {
             //System.out.println(super);  竟然编译都不让通过
               System.out.println(this); //正确,调用toString()方法
               System.out.println(super.i);
               System.out.println(this.i);
               super.test();
       }  

        public static void main(String[] args) {
               new Test().test();
        }
}  

看出什么了?没有。没有就对了,反汇编。

public void test();
   Code:
   0:   getstatic   #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   3:   aload_0
   4:   invokevirtual   #4; //Method java/io/PrintStream.println:(Ljava/lang/Object;)V
   7:   getstatic   #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   10:  aload_0
   11:  getfield    #5; //Field Base.i:I
   14:  invokevirtual   #6; //Method java/io/PrintStream.println:(I)V
   17:  getstatic   #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   20:  aload_0
   21:  getfield    #2; //Field i:I
   24:  invokevirtual   #6; //Method java/io/PrintStream.println:(I)V
   27:  aload_0
   28:  invokespecial   #7; //Method Base.test:()V
   31:  return  

public static void main(java.lang.String[]);
   Code:
   0:   new #8; //class Test
   3:   dup
   4:   invokespecial   #9; //Method "<init>":()V
   7:   invokevirtual   #10; //Method test:()V
   10:  return  

注意那几个aload_0, 这可是获取this啊(前面交代了,this是实例方法的第一个参数,第一个参数当然位置是0了)!那访问this.i和super.i的差别在什么地方呢?在这个i上。

10: aload_0
           11: getfield #5; //Field Base.i:I

这是获取Base定义的i

20: aload_0
            21: getfield #2; //Field i:I

这是获取Test中定义的i

简单的说,Base.i和i就是当前对象中两个名字不同的字段。

有人要质疑了:瞎掰吧!如果super和this是同一个对象的引用,那super调用方法的时候不又多态了 ?这的确是个问题。不过Java的设计者考虑 到这个问题了,在jvm中提供了不同的调用方法的指令,分别是invokevirtual、invokespecial、 invokeinterface、invokedynamic等 。通过super调用实例方法时,用的就是invokespecial 指令。当 然,java的设计者们也不是一开始就考虑这么多的,invokespecial这个指令就是后来才加上的。事后诸葛亮?呵呵。

再看一下上面的反编译代码:

27: aload_0
            28: invokespecial #7; //Method Base.test:()V

//用invokespecial指令避免此时发生多态调用

而多态方法调用时用什么指令呢?

看main方法是如何调用test方法的吧:public static void main(java.lang.String[]);

7: invokevirtual #10; //Method test:()V

为什么叫virtual呢?C++里有个叫做虚函数的东西,难道……

其实我们可以形象的说明一下,假设有两个变量_super和_this。

Test _this = new Test();

Super _super = _this;

这样就能更清晰的区分this和super了吧!

正确的理解this 和 super的更多相关文章

  1. 如何正确的理解和解决 ORA-01843:not a valid month

    今天码代码的时候遇到了这个问题,因为oracle用的比较少,所在查询了一下. 顿时傻眼,有很多的贴子说是因为nls_date_language的问题,还要改会话级的NLS_DATE_LANGUAGE设 ...

  2. 理解python的super

    def super(cls, inst): mro = inst.__class__.mro() return mro[mro.index(cls) + 1] super做了这些事 摘自:https: ...

  3. 怎样正确的理解和解决 ORA-01843:not a valid month

    今天码代码的时候遇到了这个问题,由于oracle用的比較少,所在查询了一下. 顿时傻眼,有非常多的贴子说是由于nls_date_language的问题,还要改会话级的NLS_DATE_LANGUAGE ...

  4. TCP三次握手的seq和ack号的【正确】理解

    1 理论知识 先上一张图,TCP/IP详解第18章的这张图描述了一个正常的三次握手和四次挥手的状态迁移,以及seq.ack序号的变化. 基本状态看图就能了解,本文主要围绕序号的变化进行讲解. 1)se ...

  5. JVM如何理解Java泛型类

    //泛型代码 public class Pair<T>{ private T first=null; private T second=null; public Pair(T fir,T  ...

  6. (转载)新手如何正确理解GitHub中“PR(pull request)”中的意思

    我从知乎看到的两个答案,分别从实际意义以及语言学角度告诉你改怎么理解PR,很简洁,这个理解非常棒,会解决新手刚看到PR(pull request)这个词时的困惑.   实际意义:   有一个仓库,叫R ...

  7. iOS 中self和super如何理解?

    或许你理解self和super都是指的是类的对象   self指的是本类的对象,而super指的是父类的对象,但是事实情况呢,可能有些和你想象的不一样? 简单看下下面例子: @interface Pe ...

  8. IM开发基础知识补课:正确理解前置HTTP SSO单点登陆接口的原理

    1.前言 一个安全的信息系统,合法身份检查是必须环节.尤其IM这种以“人”为中心的社交体系,身份认证更是必不可少. 一些PC时代小型IM系统中,身份认证可能直接做到长连接中(也就是整个IM系统都是以长 ...

  9. IM开发基础知识补课(五):通俗易懂,正确理解并用好MQ消息队列

    1.引言 消息是互联网信息的一种表现形式,是人利用计算机进行信息传递的有效载体,比如即时通讯网坛友最熟悉的即时通讯消息就是其具体的表现形式之一. 消息从发送者到接收者的典型传递方式有两种: 1)一种我 ...

随机推荐

  1. mozilla css developer center

    https://developer.mozilla.org/en-US/docs/Web/CSS

  2. Crawling is going on - Alpha版本测试报告

    [Crawling is going on - Alpha版本] 测 试 报 告 文件状态: [] 草稿 [√] 正式发布 [] 正在修改 报告编号: 当前版本: 1.0.2 编写人: 周萱.林谋武. ...

  3. canvas圆环进度

    CSS: <div class="circle"> <p><span id="loadedNum">0</span&g ...

  4. jekyll : 使用github托管你的博客

    使用github托管你的博客 效果: http://wuya1234.github.io/blog/2013/11/09/start-github-blog/ 样式神马的还没整 电脑系统 我使用的是m ...

  5. python 数据结构-字典

    原文地址:http://docs.pythontab.com/python/python3.4/datastructures.html#tut-tuples 理解字典的最佳方式是把它看做无序的键: 值 ...

  6. 看我是一只IT小小鸟有感

    当我看了<我是一只IT小小鸟>后,有许多的感想.就像许多作者一样,在接触计算机这个专业时都有许多的抱怨,对这个专业的不了解,对这个专业不知道从何学起有深深的无助感,对这个专业在未来的发展有 ...

  7. Yandex.Algorithm 2011 Round 1 D. Sum of Medians 线段树

    题目链接: Sum of Medians Time Limit:3000MSMemory Limit:262144KB 问题描述 In one well-known algorithm of find ...

  8. EF 更新条目时出错。有关详细信息,请参见内部异常。

    现象:使用EF新增记录时,一直报上述异常,网上说是值为空.主键外键未设等原因导致,但是改正这些情况下问题依然 解决过程:异常中有一句(请参见内部异常),一直都没有当回事,后来实在没办法就静下心来看了看 ...

  9. 设计模式之享元模式(FlyWeight)

    #include <iostream> #include <string> #include <list> #include <vector> usin ...

  10. 【BZOJ】【1597】【USACO 2008 Mar】土地购买

    DP/斜率优化 Orz Hzwer…… 想到排序了,但没想到其实可以将序列转化为x递增且y递减的序列……因为x是递增的,若y[i]>y[i-1]那么第i-1个就足够小……以至于可以在搞定第 i ...