首页 > 代码库 > 阅读《effective java-第17条》遇到的问题解决与分享

阅读《effective java-第17条》遇到的问题解决与分享

问题背景

  最近这2天准备重新看一遍《effective java》,发现这些经典的书籍真的是看一遍又有一遍的感受。也越来越觉的学习的过程是一个重复的过程。这次遇到的问题是在第17条中看到的,看了蛮久都没有看懂视例代码。第17条的内容是要么为继承而设计,并提供文档说明,要么就禁止继承在其中有一段示例构造器决不能调用可被覆盖的方法代码如下

父类代码

package com.sitech.test;/** * effect of java  * @author liaowp * */public class Super {        public Super(){        overrideMe();    }        public void overrideMe(){    }}

子类代码

package com.sitech.test;import java.util.Date;/** * effect of java * @author liaowp * */public class Sub extends Super{    private final Date date;        Sub(){        date =new Date();    }        @Override    public void overrideMe(){        System.out.println("时间"+date);    }        public static void main(String[] args) {        Sub sub = new Sub();        sub.overrideMe();    }}

输出结果:

技术分享

  你可能会期待这个程序会打印出日期两次,但是它第一次打印的是null,因为overrideMe方法被super构造器调用的时候,构造器Sub还没有机会初始化date域。注意,这个程序观察到的final域处于2种不同的状态。还要注意,如果overrideMe已经调用了date中的任何方法,当Super改造器调用overrideMe的时候,调用就会抛出NullPointerException异常原文

问题提出 

   看到很多的人看到上面的代码都会知道结果,但是我当时却卡在了为啥父类构造方法调用overrideMe是子类的?瞬间懵了,当时没有理解过来。脑子转不过来了,只能查查资料了,但是不知道怎么搜索。于是只能自己瞎折腾一下。于是先把代码加上打印:

技术分享

问题解决 

  确实是调用的子类的overrideMe方法,还是不太懂。只能debug了,在debug之前我又在overrideMe方法加上了一个this,我想看看this代表的是谁。

技术分享

技术分享

  debug之后可以看到this指的是Sub,date是空的,这样知道就知道父类的构造方法中调用的overrideMe是那里的,所以也就明白了父类为啥调用的是子类的方法了。main里面就是实例化了子类,this代表的就是子类,所以父类中构造方法中调用overrideMe的就是子类的overrideMe。完美

问题总结与分享

  人有时候容易思维定势,导致有时候别人很容易看出的问题自己一直懵在里面出不来,这个问题对于我来说就是这样的额,自己当时怎么都搞不懂,固定在一个地方了,问了朋友问了导师,瞬间就想清楚了在经过自己的验证,问题就是找到答案了。虽然固定在那是痛苦的,不过找到问题答案就是开心的。

  最近一直看资料看书,看到一段个人觉的非常不错的代码的分享给所有人。我们经常会看到字符串倒转的面试题,有很多实现方式,我就把自己看到的比较好的方式分享出来,这个方式也感觉跟我上面那个思维问题有点类似的感觉,个人感觉遇到字符串的倒转的问题的时候,很多会想到循环什么的。这个方法却出乎意料。代码如下:

 public static String reverse(String originStr) {            if(originStr == null || originStr.length() <= 1)                 return originStr;            return reverse(originStr.substring(1)) + originStr.charAt(0);//截取前面的补到后面去}

阅读《effective java-第17条》遇到的问题解决与分享