0%
这是一片思考的空间 -- arthinking
Spring 重构&代码整洁之道 软件设计 JVM 并发编程 数据结构与算法 分布式 存储 网络 微服务 设计模式
Java技术栈 - 涉及Java技术体系

Java笔记 - toString方法 无意识的递归

一个类继承了Object的toString()方法,并重写了toString()方法,然后在toString中使用了this造成了无意识的递归。

1、toString()方法:

我们知道, 每一个类都是继承自Object的,所以都继承了Object的toString()方法, 打印一个对象的时候,会调用其toString()方法,默认的toString是打印该对象的内存地址。

2、容器类的toString()方法:

容器类也都有toString()方法,并且覆盖了该方法,使得它生成的String结果能够表达容器自身,以及容器所包含的对象,ArrayList.toSting()方法是这样的:遍历ArrayList中的所有对象,调用每个元素上的toString()方法,下面是其源码:

public String toString() {
Iterator i = iterator();
if (! i.hasNext())
return "[]";

StringBuilder sb = new StringBuilder();
sb.append('[');
for (;;) {
    E e = i.next();
    sb.append(e == this ? "(this Collection)" : e);
    if (! i.hasNext())
    return sb.append(']').toString();
    sb.append(", ");
}
}

看下这段代码

public class ToString {
public static void main(String[] args){
User user1 = new User("Jason", "123");
User user2 = new User("arthinking", "123");
ArrayList users = new ArrayList();
users.add(user1);
users.add(user2);
System.out.println(users);
}
}

class User {
private String name;
private String id;
public User(String name, String id){
this.name = name;
this.id = id;
}
}

得到的结果为:

[com.itzhai.test.User@6e1408, com.itzhai.test.User@e53108]

如结果所示,按照ArrayList的toString()方法实现,打印出了每一个元素。我们再重写下User类的toString()方法:

@Override
public String toString() {
return this.name + ":" + this.id;
}

执行结果,每一个User元素也会调用其toString()方法:

[Jason:123, arthinking:123]

3、无意识的递归:

现在修改下程序,把User的toString()方法改为如下:

@Override
public String toString() {
return "address: " + this;
}

再次执行程序,会得到如下的异常:

Exception in thread "main" java.lang.StackOverflowError
at java.lang.AbstractStringBuilder.(Unknown Source)
at java.lang.StringBuilder.(Unknown Source)
at com.itzhai.test.User.toString(ToString.java:31)
at java.lang.String.valueOf(Unknown Source)
at java.lang.StringBuilder.append(Unknown Source)

这是递归调用出现的错误导致的。当我们使用"+"连接"address"和this时,编译器试着将this转换成一个String,这种转换直接调用了this上的toString方法,于是出现了递归调用。

解决方法:

其实解决方法也很简单,就是通过super.toString()显示的去调用父类的toString()方法,从而避免编译器调用了

@Override
public String toString() {
return "address: " + super.toString();
}

得到如下结果:

[address: com.itzhai.test.User@6e1408, address: com.itzhai.test.User@e53108]

欢迎关注我的其它发布渠道

订阅IT宅
内功修炼
Java技术栈
Java架构杂谈是IT宅精品文章公众号,欢迎订阅:
📄 网络基础知识:两万字长文50+张趣图带你领悟网络编程的内功心法 📄 HTTP发展史:三万长文50+趣图带你领悟web编程的内功心法 📄 HTTP/1.1:可扩展,可靠性,请求应答,无状态,明文传输 📄 HTTP/1.1报文详解:Method,URI,URL,消息头,消息体,状态行 📄 HTTP常用请求头大揭秘 📄 HTTPS:网络安全攻坚战 📄 HTTP/2:网络安全传输的快车道 📄 HTTP/3:让传输效率再一次起飞 📄 高性能网络编程:图解Socket核心内幕以及五大IO模型 📄 高性能网络编程:三分钟短文快速了解信号驱动式IO 📄 高性能网络编程:彻底弄懂IO复用 - IO处理杀手锏,带您深入了解select,poll,epoll 📄 高性能网络编程:异步IO:新时代的IO处理利器 📄 高性能网络编程:网络编程范式 - 高性能服务器就这么回事 📄 高性能网络编程:性能追击 - 万字长文30+图揭秘8大主流服务器程序线程模型
📄 Java内存模型:如果有人给你撕逼Java内存模型,就把这些问题甩给他 📄 一文带你彻底理解同步和锁的本质(干货) 📄 AQS与并发包中锁的通用实现 📄 ReentrantLock介绍与使用 📄 ReentrantReadWriteLock介绍与使用 📄 ReentrantLock的Condition原理解析 📄 如何优雅的中断线程 📄 如何优雅的挂起线程 📄 图解几个好玩的并发辅助工具类 📄 图解BlockingQueue阻塞队列
📄 消息队列那么多,为什么建议深入了解下RabbitMQ? 📄 高并发异步解耦利器:RocketMQ究竟强在哪里? 📄 Kafka必知必会18问:30+图带您看透Kafka
📄 洞悉MySQL底层架构:游走在缓冲与磁盘之间 📄 SQL运行内幕:从执行原理看调优的本质 📄 洞悉Redis技术内幕:缓存,数据结构,并发,集群与算法