正确的理解this 和 super
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的更多相关文章
- 如何正确的理解和解决 ORA-01843:not a valid month
今天码代码的时候遇到了这个问题,因为oracle用的比较少,所在查询了一下. 顿时傻眼,有很多的贴子说是因为nls_date_language的问题,还要改会话级的NLS_DATE_LANGUAGE设 ...
- 理解python的super
def super(cls, inst): mro = inst.__class__.mro() return mro[mro.index(cls) + 1] super做了这些事 摘自:https: ...
- 怎样正确的理解和解决 ORA-01843:not a valid month
今天码代码的时候遇到了这个问题,由于oracle用的比較少,所在查询了一下. 顿时傻眼,有非常多的贴子说是由于nls_date_language的问题,还要改会话级的NLS_DATE_LANGUAGE ...
- TCP三次握手的seq和ack号的【正确】理解
1 理论知识 先上一张图,TCP/IP详解第18章的这张图描述了一个正常的三次握手和四次挥手的状态迁移,以及seq.ack序号的变化. 基本状态看图就能了解,本文主要围绕序号的变化进行讲解. 1)se ...
- JVM如何理解Java泛型类
//泛型代码 public class Pair<T>{ private T first=null; private T second=null; public Pair(T fir,T ...
- (转载)新手如何正确理解GitHub中“PR(pull request)”中的意思
我从知乎看到的两个答案,分别从实际意义以及语言学角度告诉你改怎么理解PR,很简洁,这个理解非常棒,会解决新手刚看到PR(pull request)这个词时的困惑. 实际意义: 有一个仓库,叫R ...
- iOS 中self和super如何理解?
或许你理解self和super都是指的是类的对象 self指的是本类的对象,而super指的是父类的对象,但是事实情况呢,可能有些和你想象的不一样? 简单看下下面例子: @interface Pe ...
- IM开发基础知识补课:正确理解前置HTTP SSO单点登陆接口的原理
1.前言 一个安全的信息系统,合法身份检查是必须环节.尤其IM这种以“人”为中心的社交体系,身份认证更是必不可少. 一些PC时代小型IM系统中,身份认证可能直接做到长连接中(也就是整个IM系统都是以长 ...
- IM开发基础知识补课(五):通俗易懂,正确理解并用好MQ消息队列
1.引言 消息是互联网信息的一种表现形式,是人利用计算机进行信息传递的有效载体,比如即时通讯网坛友最熟悉的即时通讯消息就是其具体的表现形式之一. 消息从发送者到接收者的典型传递方式有两种: 1)一种我 ...
随机推荐
- C++设计模式——代理模式
前言 青春总是那样,逝去了才开始回味:大学生活也是在不经意间就溜走了,现在上班的时候,偶尔还会怀念大学时,大家在一起玩游戏的时光.大学喜欢玩游戏,但是可悲的校园网,速度能把人逼疯了:还好,后来搞了一个 ...
- java并发编程:如何创建线程
原文:http://www.cnblogs.com/dolphin0520/p/3913517.html 一.Java中关于应用程序和进程相关的概念 在Java中,一个应用程序对应着一个JVM实例(也 ...
- 01.JSP基础语法
本章主要讲解Java Web与JSP的入门内容,适合有JSP或Java Web基础的读者学习. 1.Web应用与web.xml文件 (1)Java Web应用程序的结构 Java We ...
- Applicationpoolidentity 好有趣哦
前言: 在服务器上搭建一个网站 Windows Server2008 R2 +IIS7. 应用程序池默认选择ApplicationPoolIdentity.为了适应分布式需求,所以将SqlServer ...
- 【BZOJ】【3207】花神的嘲讽计划 I
字符串Hash+可持久化线段树 好神奇的转化…… 蒟蒻一开始还去想AC自动机去了……然而由于a[i]的范围是小于等于n,怎么也想不出一个时间复杂度合理的方法 膜拜了题解0.0原来是字符串Hash! 首 ...
- 【BZOJ】【3907】网格
组合数学/python 3907: 网格 Time Limit: 1 Sec Memory Limit: 256 MBSubmit: 162 Solved: 76[Submit][Status][ ...
- NYOJ-32 组合数 AC 分类: NYOJ 2013-12-30 07:42 189人阅读 评论(0) 收藏
#include<stdio.h> int num[100]; int pnum(int n,int v); int mv=0; int main(){ int n,v; scanf(&q ...
- JS数组2(冒泡排列、数组里面查找数据)
数组 一.冒泡排列 对数组attr = [1,8,6,4,5,3,7,2,9]进行由大到小排列,用冒泡排列的方法排列时,会对数组进行比较互换.如果前一个数字较大,这2个元素排列方式不变,如果后一个元素 ...
- php随机数怎么获取?一个简单的函数就能生成
小美女建了一个站,有些页面相似度比较高,想添加一些字段来实现差异化,比如用php随机数生成从10到100之间随机一个数字.其实会php的朋友几十个字符就能实现了,如下代码所示,简单吧?10代表最小值, ...
- javascript实现数据结构与算法系列:线性表的静态单链表存储结构
有时可借用一维数组来描述线性链表,这就是线性表的静态单链表存储结构. 在静态链表中,数组的一个分量表示一个结点,同时用游标(cur)代替指针指示结点在数组中的相对位置.数组的第0分量可看成头结点,其指 ...