先看一段很简单的java代码:

  1. toString()/**
  2. * @author jeffwong
  3. */
  4. public class InfiniteRecursion {
  5. public String toString(){
  6. return "InfiniteRecursion address "+ this + "\n";
  7. }
  8.  
  9. public static void main(String[] args) {
  10. InfiniteRecursion obj = new InfiniteRecursion();
  11. System.out.println(obj);
  12. }
  13. }

运行后,我们看到了一串异常信息,StackOverflow了:

爆栈的原因通常是因为递归或者无限循环,上面的代码造成堆栈溢出的直接原因是递归,下面来分析一下造成递归的原因:

定义类InfiniteRecursion,本来希望通过toString方法打印出当前对象的内存地址,获取当前对象当然会想到this关键字。在toString方法内,Java编译器发现一个字符串后面跟着一个加号,加号后面的对象不是String类型,所以编译器试着自动将this转换为字符串类型,也就是调用this.toString()方法,偏偏this所指代的这个对象重新实现了toString方法,于是造成循环不断地调用toString方法,这样就发生了递归调用。

上面这种toString()被Bruce Eckel总结称为无意识的递归,在.NET当中同样适用(重写ToString方法),类似代码我就不贴了。

这种无意识的递归说到底还是使用类继承不当造成的。也许你会说其实这根本没有什么技术难度的,类继承用对了不就行了吗,不就是继承自Object的区区一个ToString方法吗?

如果你手上维护的代码是一个基类代码丰富,继承链很深,含有大量包装类,类调用关系复杂……用对就行了,说起来未免太容易。再说一个由类继承引发的血案的真实案例:

用户反馈我司某重要业务部门的线上系统某页面无法正常打开,直观效果就是页面一直loading迟迟打不开。开发人员费了很大力气据说还动用了Windbg神器和传说中的架构师才排查到是一个页面方法调用的问题,为了说明问题我们暂时命名该页面为A,页面A继承自自定义基类页面B,然后出现问题的原因大致分析如下:

1、A页面某方法func1触发调用了基类B的另一个方法func2;

2、B中的方法func2执行的时候,发现被A重写了,所以就去执行A重写的方法func2;

3、A重写的func2内部又调用了func1,这样就重新触发A的func1的执行。

这样,一个神不知鬼不觉惊天地泣鬼神的无意识递归就形成了。

饱满而真实的类继承的副作用的经典案例介绍到此结束。

明天,你还会用类继承吗,你还敢用类继承吗,你还忍心用类继承吗?

参考:<<Thinking in Java>>

由Java中toString()方法引发的无意识的递归想到的的更多相关文章

  1. 自己(转)JAVA中toString方法的作用

    JAVA中toString方法的作用 因为它是Object里面已经有了的方法,而所有类都是继承Object,所以“所有对象都有这个方法”. 它通常只是为了方便输出,比如System.out.print ...

  2. JAVA中toString方法的作用

    因为它是Object里面已经有了的方法,而所有类都是继承Object,所以“所有对象都有这个方法”. 它通常只是为了方便输出,比如System.out.println(xx),括号里面的“xx”如果不 ...

  3. JAVA中toString方法的作用(转)

    因为它是Object里面已经有了的方法,而所有类都是继承Object,所以“所有对象都有这个方法”. 它通常只是为了方便输出,比如System.out.println(xx),括号里面的“xx”如果不 ...

  4. JAVA中toString方法

    因为它是Object里面已经有了的方法,而所有类都是继承Object,所以"所有对象都有这个方法". 它通常只是为了方便输出,比如System.out.println(xx),括号 ...

  5. Java中的方法应用

    一.如何定义java中的方法 所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块. 语法: 1. 访问修饰符:方法允许被访问的权限范围, 可以是 public.protected.priv ...

  6. Java中的方法(形参及实参)return返回类型

    如何定义 Java 中的方法 所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块. 一般情况下,定义一个方法的语法是: 其中: 1. 访问修饰符:方法允许被访问的权限范围, 可以是 pub ...

  7. java中的方法method

    java中的方法必须存在于类class里,不能独立存在.类是描述具有某种特征的事物,方法则是这类 事物具有的某种功能,通过调用方法可以实现某种特定的功能.方法名一般以小写的动词开头. 例: publi ...

  8. Java09-java语法基础(八)java中的方法

    Java09-java语法基础(八)java中的方法 一.方法(函数/过程):是一个程序块,可以完成某种功能 1.java中方法的定义格式 [访问控制修饰符]  返回值类型  方法名(参数列表){ 方 ...

  9. Java学习笔记十一:Java中的方法

    Java中的方法 一:什么是方法: 所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块. 学过C语言或者其他语言的应该都知道函数这个东西,在Java中,其实方法就是函数,只不过叫法不同,在 ...

随机推荐

  1. 分享一个 jquery serializeArray()序列化方法

    http://www.365mini.com/page/jquery-serializearray.htm http://www.365mini.com/diy.php?f=jquery-serial ...

  2. http://10.11.112.12:8000/homePage.jsp?loginMode=JS

    sdsad http://svn.paic.com.cn/svn/mis_pac/branches/MIS-PAC5.22.0

  3. storyboard xib下label怎么自适应宽度高度

    先看需求:两个Label,要求蓝色的label紧跟在红色的label文字后面  ok首选正常添加约束 红色的Label添加宽度,高度,左边,上边约束 蓝色的Label添加宽度,高度,左边,和红色的水平 ...

  4. ftl文件格式化jsp形式显示

    通过myeclipse设置ftl的编辑器为jsp的编辑器,达到效果. Myeclipse->windows->preferences 1\ 2\ 3\ 完成后即可.

  5. 各种 starter poms (启动器)

    starter包含了搭建项目,快速运行所需的依赖.它是一个依赖关系描述符的集合.当应用需要一种spring的服务时,不需要粘贴拷贝大量的依赖关系描述符.例如想在spring中使用redis,只需要在项 ...

  6. 获取datagrid选择行

    var rows = $('#dg').datagrid('getChecked');     var ids = '';  for (var i = 0; i < rows.length; i ...

  7. 6.HotSpot垃圾收集器

    HotSpot JVM收集器 上面有7中收集器,分为两块,上面为新生代收集器,下面是老年代收集器.如果两个收集器之间存在连线,就说明它们可以搭配使用. 并发和并行 先解释下什么是垃圾收集器的上下文语境 ...

  8. Android Log介绍

    android.util.Log常用的方法有以下5个:Log.v() ,Log.d() ,Log.i() ,Log.w() ,Log.e() .按照日志级别从高到低为ERROR, WARN, INFO ...

  9. [转]Sql server2005中如何格式化时间日期

    ) -- mon dd yyyy hh:mmAM (or PM) ) -- mm/dd/yyyy - 10/02/2008 ) -- yyyy.mm.dd -- 2008.10.02 ) -- dd/ ...

  10. 6本Android开发必备图书

    学习一样新事物或许有多种方式,报培训班,看视频,向高手请教等等,但一本好书往往可以让你少走很多弯路,事半功倍.今天与大家分享一些Android开发值得一读的书籍,希望对新手安卓开发者们有所帮助. 1. ...