从java构造函数看出点什么

13 篇文章 0 订阅
1 篇文章 0 订阅

我们先建立一个父类,名为Father.java,类中代码如下:

public class Father{}

再建立一个子类,名为Son.java,类中代码如下:

public class Son extends Father {}

我们知道,如果在类中没有声明构造函数,javac编译时,会自动的为我们加上一个公开的无参构造函数。

Windows7 cmd下JDK1.8.0_151环境,我们使用【javac Son.java】命令编译Son.java代码,javac会“聪明的”将Father也编译了。

于是我们可以得到Son.class和Father.class两个字节码文件。

我们先看看Father.class反编译结果,使用【javap -c Father.class】命令查看:

果不其然,javac自动的为我们添加了一个无参的构造函数。可是,细心的读者一定会奇怪的发现,这个java.lang.Object是干吗的?还用到了一个字节码指令invokespecial。它们都是干吗的? 别着急,我们先看看Son.class的反编译结果。

使用命令【Javap -c Son.class】看看Son.class反编译的结果:

同Father.class反编译结果类似,javac自动为Son.class增加了一个无参构造函数。

重点在这个构造函数里,Method Father."<init>":()V语句。是不是与Father.class中构造函数很类似,只是成了Method java/lang/Object."<init>":()V。

根据上面这些信息,我们开始思考。

同是invokespecial指令,既然Son是继承自Father,那么Father是不是以此推理,继承自java.lang.Object呢?可我们在Father类的定义中并没有声明它继承Object类啊?原因在于,java中默认所有“没有声明继承其他类的”的类都是默认继承java.lang.Object类。至于为什么要继承Object,我会在其他的文章里讲述。这里读者只需要记住这个特点就行了。

我们改造一下Father和Son的源文件,显示声明默认构造函数,看看会发生什么。

public class Father {
    public Father() {}
}
public class Son extends Father {
    public Son() {}
}

还是使用javac编译,得到Son.class和Father.class,这次我将两个类反编译结果同时展示出来:

是不是和上面的结果一模一样。

我把Son.java文件,再改造一下,改成这样:

public class Son extends Father() {
    public Son() {
        super();
    }
}

再次编译Son.java,得到新的Son.class和Father.class文件,展示如下:

还是一样。我们把Son.java代码改造一下,这样写:

public class Son extends Father {
    public Son() {
        new Father();
    }
}

然后编译Son.java文件,再反编译Son.class得到如下结果:

同上面的图片对比,new是分水岭,我们知道了Method Father."<init>":()V调用的是Father的构造函数。并且在这里调用了两次。由此我们知道了,子类执行构造函数时,会默认执行父类构造函数,并且首先执行父类默认构造函数。

【总结】:子类执行构造函数,会默认执行父类构造函数。

【思考】:如果父类中有多个构造函数呢?如果子类中有多个构造函数呢,怎么调用?

【备注】:限于篇幅,还是没有讲super和invokespecial指令是干嘛的,我会在其他文章中专门来讲解。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
/** * 一、BufferedReader类 public class BufferedReader extends Reader * 从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。 可以指定缓冲区的大小,或者可使用默认的大小。大多数情况下,默认值足够大。 * 通常,Reader 所作的每个读取请求都会导致对底层字符或字节流进行相应的读取请求。因此,建议用 BufferedReader包装所有其 read() * 操作可能开销很高的 Reader(如 FileReader和 InputStreamReader)。 * BufferedReader流能够读取文本行,通过向BufferedReader传递一个Reader对象 * ,来创建一个BufferedReader对象,之所以这样做是因为FileReader没有提供读取文本行的功能. * * 二、InputStreamReader类 * * InputStreamReader 将字节流转换为字符流。是字节流通向字符流的桥梁。如果不指定字符集编码,该解码过程将使用平台默认的字符编码,如:GBK。 * * 构造方法: * * InputStreamReader isr = new InputStreamReader(InputStream * in);//构造一个默认编码集的InputStreamReader类 * * InputStreamReader isr = new InputStreamReader(InputStream in,String * charsetName);//构造一个指定编码集的InputStreamReader类。 * * 参数 in对象通过 InputStream in = System.in;获得。//读取键盘上的数据。 * * 或者 InputStream in = new FileInputStream(String fileName);//读取文件中的数据。可以看出 * FileInputStream 为InputStream的子类。 * * 主要方法:int read();//读取单个字符。 int read(char []cbuf);//将读取到的字符存到数组中。返回读取的字符数。 * * 三、FileWriter(少量文字) 和 BufferedWriter(大量文字)实现简单文件写操作 * @author hulk */

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值