ITEM 12: 重写 ToString

ITEM 12: ALWAYS OVERRIDE TOSTRING
  Object 提供 toString 方法的默认实现,但它返回的字符串通常不是类的用户想要看到的。它由类名后跟“at”符号(@)和散列代码的无符号十六进制表示形式组成,例如 PhoneNumber@163b91。toString的一般契约规定,返回的字符串应该是“简洁但信息丰富的表示形式,便于阅读”。虽然可以认为PhoneNumber@163b91简洁易懂,但与707-867-5309相比,它的信息量还是过少。
  toString 的契约提供了一个很好的解决方案: “建议所有子类重写这个方法。”。
  虽然它没有遵守 equals 和 hashCode 契约那么重要,但是提供一个好的toString 实现会使您的类更易于使用,并且使使用该类的系统更容易调试。当对象传递给println、printf、字符串连接操作符或 assert ,或由调试器打印时,toString方法将自动被调用。即使您从来没有调用对象上的 toString 方法,其他人可能会这样做。例如,输出一个错误日志,信息中包含一个引用了没有覆盖toString方法的类,那么此时我们输出的消息可能几乎毫无用处。如果您已经为 PhoneNumber 提供了一个好的 toString 方法,那么生成一个有用的诊断消息就像这样简单:
System.out.println("Failed to connect to " + phoneNumber);
  无论您是否覆盖 toString,程序员都将以这种方式生成诊断消息,但是除非您这样做,否则这些消息将毫无用处。提供好的 toString 方法的好处不仅限于类的实例,还包括包含对这些实例的引用的对象,尤其是集合。在打印地图时,{Jenny=PhoneNumber@163b91} 或 {Jenny=707-867-5309},您更愿意看到哪个?事实上,toString 方法应该返回对象中包含的所有用户可能感兴趣的信息,如电话号码示例所示。但如果对象很大,或者它包含不利于字符串表示的字段,那么这可能是不切实际的目标。在这种情况下,toString 应该返回 Manhattan residential phone directory (1487536 listings) 或 Thread[main,5,main]。理想情况下,字符串应该是自解释的。如果没有在字符串表示中包含对象的所有有用信息,可能会产生像下面这样一个让人摸不着头脑的日志:
Assertion failure: expected {abc, 123}, but was {abc, 123}.
  实现toString方法时必须做出的一个重要决定是,是否在文档中指定返回值的格式。建议对值类(如电话号码或矩阵)这样做。指定格式的好处是,它可以作为对象的标准、明确、人类可读的表示形式。这种表示可以用于输入和输出,也可以用于持久化可读数据对象,比如CSV文件。如果指定了格式,通常最好提供一个匹配的静态工厂或构造函数,以便程序员能够轻松地在对象及其字符串表示形式之间来回转换。Java平台库中的许多值类都采用这种方法,包括 BigInteger、BigDecimal 和大多数装箱的基元类。
  指定 toString 返回值格式的缺点是,一旦指定了它,就必须一直使用它,假设您的类被广泛使用。程序员将编写代码来解析表示,生成表示,并将其嵌入到持久数据中。如果您在将来的版本中更改表示形式,您将破坏他们的代码和数据,而他们将会咆哮。通过选择不指定格式,您可以保留在后续版本中添加信息或改进格式的灵活性。无论您是否决定指定格式,您都应该清楚地记录您的意图。如果指定了格式,就应该精确地指定。例如,这里有一个 toString 方法与item 11中的 PhoneNumber 类一起使用:

/**
* Returns the string representation of this phone number.
* The string consists of twelve characters whose format is
* "XXX-YYY-ZZZZ", where XXX is the area code, YYY is the
* prefix, and ZZZZ is the line number. Each of the capital
* letters represents a single decimal digit. *
* If any of the three parts of this phone number is too small
* to fill up its field, the field is padded with leading zeros.
* For example, if the value of the line number is 123, the last
* four characters of the string representation will be "0123". 
*/
@Override 
public String toString() {
  return String.format("%03d-%03d-%04d", areaCode, prefix, lineNum); 
}

  如果您决定不指定格式,文档注释应该如下所示:

/**
* Returns a brief description of this potion. The exact details
* of the representation are unspecified and subject to change,
* but the following may be regarded as typical: *
* "[Potion #9: type=love, smell=turpentine, look=india ink]" 
*/
@Override public String toString() { ... }

  在阅读了注释之后,当格式发生更改时,生成依赖于格式细节的代码或持久数据的程序员只能怪他们自己。无论是否指定格式,都要提供对toString返回的值中包含的信息的编程访问。例如,PhoneNumber 类应该包含区号、前缀和行号的访问器。如果做不到这一点,就会迫使需要此信息的程序员解析字符串。除了降低性能和为程序员做不必要的工作之外,这个过程还容易出错,如果更改格式,会导致脆弱的系统崩溃。由于无法提供访问器,您可以将字符串格式转换为事实上的API,即使您已经指定它可能会发生更改。
  在静态实用程序类中编写 toString 方法毫无意义,也不应该在大多数 enum 类型中编写 toString 方法,因为Java为您提供了一个非常好的方法。但是,应该对抽象类编写 toString 方法,这样它的子类能使用公共的字符串表示方法。例如,大多数集合实现上的 toString 方法都是从抽象集合类继承的。
  item 10 中讨论的谷歌开源自动值工具将为您生成 toString 方法,大多数 IDE 也有这样的功能。这些方法可以很好地告诉您每个字段的内容,但并不专门针对类的含义。例如,为我们的 PhoneNumber 类使用自动生成的toString方法是不合适的(因为电话号码有一个标准的字符串表示),但是对于我们的 Potion 类来说,这是完全可以接受的。也就是说,自动生成的 toString 方法要比从 Object 继承的方法好得多,后者对对象的值一无所知。
  重述一下,除非超类已经这样做了,否则您应该在编写的每个实例化类中重写 Object的 toString 实现。它使类更易于使用,并有助于调试。toString 方法应该以美观的格式返回对象的简洁、有用的描述。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容

  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,225评论 0 4
  • 一、基础知识:1、JVM、JRE和JDK的区别:JVM(Java Virtual Machine):java虚拟机...
    杀小贼阅读 2,373评论 0 4
  •   引用类型的值(对象)是引用类型的一个实例。   在 ECMAscript 中,引用类型是一种数据结构,用于将数...
    霜天晓阅读 1,045评论 0 1
  • 本章内容 使用对象 创建并操作数组 理解基本的 JavaScript 类型 使用基本类型和基本包装类型 引用类型的...
    闷油瓶小张阅读 677评论 0 0
  • 雨后的空气清清爽爽,小风吹得不急不躁,头顶上的月亮和星星清亮又柔和,在这样的晚上到操场上跑几圈儿,心情轻松又舒畅。...
    加加妈妈阅读 335评论 0 2