结果集 tostring

使用Java或我使用过的其他编程语言,我发现有时候可以用该语言完成某些事情,但通常不应该这样做。 通常,这些误用语言似乎无害,当开发人员首次使用它们时可能是有益的,但后来同一位开发人员或另一位开发人员遇到了相关的问题,需要克服或改变这些代价。 一个示例(也是本博客文章的主题)是使用Java中toString()调用的结果进行逻辑选择或对其内容进行解析。

在2010年,我用Java语言编写了toString()注意事项 ,当toString()方法可明确用于类时以及当它们包含该类对象的相关公共状态时,我通常会首选它。 我仍然有这种感觉。 但是,我希望toString()实现足以使人们通过记录的语句或调试器读取对象的内容,而不是要由代码或脚本解析的内容。 使用toString()方法返回的String进行任何类型的条件或逻辑处理都非常脆弱。 同样,解析toString()返回的String以获取有关实例状态的详细信息也是脆弱的。 我警告过(甚至是无意间)要求开发人员在前面提到的博客文章中解析toString()结果。

开发人员可能出于多种原因选择更改toString()的生成的String,包括将现有字段添加到以前可能未表示过的输出中,将更多数据添加到已经表示过的现有字段中,为新添加的字段添加文本,删除不再在课程中的字段的表示形式或出于美学原因更改格式。 开发人员还可以更改toString()生成的String拼写和语法问题。 如果toString()提供的String仅由人类在日志消息中分析对象的状态时使用,则除非它们删除了实质信息,否则这些更改不太可能成为问题。 但是,如果代码依赖于整个String或为某些字段解析String ,则可以通过这些类型的更改轻松地将其破坏。

出于说明目的,请考虑以下Movie类的初始版本:

package dustin.examples.strings;/*** Motion Picture, Version 1.*/
public class Movie
{private String movieTitle;public Movie(final String newMovieTitle){this.movieTitle = newMovieTitle;}public String getMovieTitle(){return this.movieTitle;}@Overridepublic String toString(){return this.movieTitle;}
}

在这个简单且有些人为的示例中,只有一个属性,因此类的toString()仅仅返回该类的单个String属性作为类的表示形式并不稀奇。

下一个代码清单包含一个不幸的决定(第22-23行),该决定基于Movie类的toString()方法的逻辑。

/*** This is a contrived class filled with some ill-advised use* of the {@link Movie#toString()} method.*/
public class FavoriteMoviesFilter
{private final static List<Movie> someFavoriteMovies;static{final ArrayList<Movie> tempMovies = new ArrayList<>();tempMovies.add(new Movie("Rear Window"));tempMovies.add(new Movie("Pink Panther"));tempMovies.add(new Movie("Ocean's Eleven"));tempMovies.add(new Movie("Ghostbusters"));tempMovies.add(new Movie("Taken"));someFavoriteMovies = Collections.unmodifiableList(tempMovies);}public static boolean isMovieFavorite(final String candidateMovieTitle){return someFavoriteMovies.stream().anyMatch(movie -> movie.toString().equals(candidateMovieTitle));}
}

尽管有多个电影共用同一标题 ,但尽管存在一些潜在的问题,但该代码似乎仍然可以工作一段时间。 但是,即使在遇到这些问题之前,如果开发人员确定他或她想将Movie.toString()表示形式的格式更改为下一个显示的内容,则可能会意识到在相等性检查中使用toString()的风险。代码清单。

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

也许更改了Movie.toString()返回值,以使提供的StringMovie类的实例相关联更加清楚。 不管进行更改的原因如何,以前列出的在影片标题上使用相等性的代码现在都已损坏。 该代码需要更改为使用contains而不是equals ,如下面的代码清单所示。

public static boolean isMovieFavorite(final String candidateMovieTitle)
{return someFavoriteMovies.stream().anyMatch(movie -> movie.toString().contains(candidateMovieTitle));
}

当意识到Movie类需要更多信息来使电影与众不同时,开发人员可以将发行年份添加到movie类。 接下来显示新的Movie类。

package dustin.examples.strings;/*** Motion Picture, Version 2.*/
public class Movie
{private String movieTitle;private int releaseYear;public Movie(final String newMovieTitle, final int newReleaseYear){this.movieTitle = newMovieTitle;this.releaseYear = newReleaseYear;}public String getMovieTitle(){return this.movieTitle;}public int getReleaseYear(){return this.releaseYear;}@Overridepublic String toString(){return "Movie: " + this.movieTitle;}
}

添加发行年份有助于区分具有相同标题的电影。 这也有助于将翻拍与原作区分开。 但是,无论电影发行的年份如何,使用Movie类查找收藏夹的代码仍将显示所有具有相同标题的电影。 换句话说,1960年版的《 海洋十一人》 ( 目前对IMDB评分为6.6 )将与2001年版的《 海洋十一人》 ( 目前对IMDB评分为7.8 )一起成为人们的最爱,尽管我更喜欢较新的版本。 同样,1988年提出为电视版后窗 (的5.6评级目前IMDB )将返回为收藏旁边的1954年版后窗 (执导的阿尔弗雷德·希区柯克 ,主演詹姆斯·斯图尔特和格蕾丝·凯莉 ,以及额定8.5目前在IMDB中 ),尽管我更喜欢旧版本。

我认为toString()实现通常应包含对象的所有公共可用细节。 但是,即使将MovietoString()方法增强为包括发行年份,客户端代码也不会基于年份进行区分,因为它仅对电影标题执行contain

@Override
public String toString()
{return "Movie: " + this.movieTitle + " (" + this.releaseYear + ")";
}

上面的代码显示了添加到MovietoString()实现中的发行年份。 下面的代码显示了如何更改客户以正确遵守发布年份。

public static boolean isMovieFavorite(final String candidateMovieTitle,final int candidateReleaseYear)
{return someFavoriteMovies.stream().anyMatch(movie ->   movie.toString().contains(candidateMovieTitle)&& movie.getReleaseYear() == candidateReleaseYear);
}

我想情况下它是一个解析一个好主意,这是很难toString()上的结果,方法或基础条件或其他逻辑toString()方法。 在我考虑的几乎所有示例中,都有更好的方法。 在上面的示例中,最好向Movie添加equals() (和hashCode() )方法,然后对Movie实例使用相等性检查,而不要使用单个属性。 如果确实需要比较各个属性(例如,在不需要对象相等且只需要一个或两个字段相等的情况下),则可以使用适当的getXXX方法。

作为一名开发人员,如果我希望类的用户(通常会最终包括我自己)不需要解析toString()结果或依赖于某个结果,则需要确保我的类使toString()提供任何有用的信息toString()可从其他易于访问且更编程友好的资源中获得,例如“获取”方法以及相等性和比较方法。 如果开发人员不想通过公共API公开某些数据,则很可能开发人员也可能真的不想在返回的toString()结果中公开数据。 Joshua Bloch ( Effective Java)以粗体强调该文本,“……提供对toString()返回的值中包含的所有信息的编程访问。”

Effective Java中 ,Bloch还包括有关toString()方法是否应具有其提供的String表示形式的公告格式的讨论。 他指出,这种表示形式(如果进行广告宣传)必须是从那时起一直使用的,如果它是广泛使用的类,则可以避免我在本文中演示的运行时中断类型。 他还建议,如果不能保证格式保持不变,则Javadoc也应包含与此相关的声明。 通常,由于Javadoc和其他注释经常比我想要的更被忽略,并且由于所宣传的toString()表示形式具有“永久性”,因此我宁愿不依赖于toString()提供客户端所需的特定格式,而是提供一种专用于客户可以调用的方法。 这使我可以灵活地在类更改时更改toString()

JDK中的示例说明了我的首选方法,还说明了将特定格式指定为toString()的早期版本的危险。 BigDecimal的toString()表示在JDK 1.4.2和Java SE 5之间进行了更改,如“ J2SE 5.0中的不兼容性(自1.4.2起) ”所述:“ J2SE 5.0 BigDecimaltoString()方法的行为与早期版本不同版本。” BigDecimal.toString()的1.4.2版本的Javadoc仅在方法概述中声明:“返回此BigDecimal的字符串表示形式。 使用Character.forDigit(int,int)提供的数字到字符的映射。 前导减号用于表示符号,小数点右边的位数用于表示刻度。 (此表示形式与(字符串)构造函数兼容。)” Java SE 5和更高版本中BigDecimal.toString()的相同方法概述文档更加详细。 这样冗长的描述,在此不再赘述。

BigDecimal.toString()是与Java SE 5改变 ,其他的方法被引入本不同String表示: toEngineeringString()和toPlainString() 。 新引入的方法toPlainString()提供了JDK 1.4.2提供的BigDecimaltoString() 。 我倾向于提供提供特定String表示形式和格式的方法,因为这些方法可以具有其名称中描述的格式的细节,并且Javadoc注释以及对类的更改和添加不会像对它们产生影响那样对这些方法产生影响一般的toString()方法。

有一些简单的类可能适合原始实现的toString()方法将一劳永逸地修复且“永远不会”改变的情况。 那些可能是解析返回字符串或在基础逻辑候选String ,但即使在这种情况下,我更喜欢提供一种具有广告和有保证的格式的另一种方法和离开toString()表示一些灵活性变化。 拥有多余的方法没什么大不了的,因为尽管它们返回相同的内容,但多余的方法可以仅仅是调用toString的单行方法。 然后,如果toString()确实发生了变化,则可以将调用方法的实现更改为以前提供的toString() ,并且该额外方法的任何用户都不会看到任何更改。

当将toString()结果解析为逻辑或基于toString()调用的结果逻辑时,最有可能在将特定方法视为客户访问特定数据的最简单方法时进行。 最好通过其他特定的公共可用方法来使数据可用,并且类和API设计人员可以通过确保toString()提供的String中甚至可能有用的任何数据也可以通过编程访问的特定替代方式来提供帮助。方法。 简而言之,我的首选是将toString()一种方法,以查看有关表示形式中实例的一般信息,该实例可能会发生更改,并为表示形式中的特定数据段提供特定的方法,这些数据的更改可能性较小且更容易以编程方式访问决策并基于可能需要特定于格式的解析的大型String进行决策。

翻译自: https://www.javacodegeeks.com/2016/05/virtues-avoiding-parsing-basing-logic-tostring-result.html

结果集 tostring

结果集 tostring_关于避免对toString()结果进行解析或基于逻辑的美德相关推荐

  1. 关于避免对toString()结果进行解析或基于逻辑的美德

    使用Java或我使用过的其他编程语言,我发现有时可以用该语言完成某些事情,但通常不应该这样做. 通常,这些误用语言似乎无害,当开发人员首次使用它们时可能有益,但后来同一位开发人员或另一位开发人员遇到了 ...

  2. python使用matplotlib对比多个模型的在训练集上的效果并使用柱状图进行可视化:基于交叉验证的性能均值(mean)和标准差(std)进行可视化分析、使用标准差信息添加误差区间条yerr

    python使用matplotlib对比多个模型的在训练集上的效果并使用柱状图进行可视化:基于交叉验证的性能均值(mean)和标准差(std)进行可视化分析.使用标准差信息添加误差区间条yerr 目录

  3. String/StringBuilder/ToString()底层代码解析( JAVA / C# )

    C# /JAVA: 字符串构建利器StringBuilder区别 前言 名词解释 1.1 示例 案例一: 不同变量赋值( = ) 案例二:相同变量赋值( = ) 案例三:变量追加赋值( += ) 1. ...

  4. elasticsearch中集群选举中的ping源码解析

    在elasticsearch在选举中,节点之间的相互投票通过ping来实现. 其中的实现类为UnicastZenPing,在其构造方法中读取配置中的discovery.zen.ping.unicast ...

  5. 平切分,分库,分表,主从,集群 数据库水平切分的实现原理解析

    第1章  引言 随着互联网应用的广泛普及,海量数据的存储和访问成为了系统设计的瓶颈问题.对于一个大型的互联网应用,每天几十亿的PV(page view)[即页面浏览量,或点击量]无疑对数据库造成了相当 ...

  6. NodeMCU连接天猫精灵第二集——开发环境、运行条件、工作代码(基于Blinker点灯科技)

    NodeMCU连接天猫精灵第一集--硬件构架(基于Blinker点灯科技) 根据上面说的硬件部分已经搭建完成了,所以现在说一下软件开发,首先要有一个共识,NodeMCU是可以用Arduino IDE开 ...

  7. 【2022集创赛】安谋科技杯三等奖:基于ARM处理器的无线SoC设计

    本篇文章是2022年第六届全国大学生集成电路创新创业大赛安谋科技杯三等奖作品分享,参加极术社区的**[有奖征集]分享你的2022集创赛作品,秀出作品风采**活动. 团队介绍 参赛单位:厦门大学 队伍名 ...

  8. 近年GDC服务器分享合集(二): 《太空工程师》中基于预测物理的多人游戏

    客户端-服务端之间的位置同步一直是游戏开发中的一道难题,特别是还涉及到复杂的物理运动时.对于这个话题,来自<太空工程师>游戏的工程师在GDC 2023上为我们带来了他们的分享--<& ...

  9. java向后兼容吗_Java向后不兼容历史的观察

    java向后兼容吗 在大多数情况下,Java是一个非常向后兼容的编程语言. 这样做的好处是,与大规模破坏兼容性相比,大型系统通常可以相对轻松的方式升级为使用Java的较新版本. 这样做的主要缺点是Ja ...

最新文章

  1. Redis初学:8(Hash类型)
  2. Computed property XXX was assigned to but it has no setter
  3. Android面试真题解析火爆全网,薪资翻倍
  4. mysql修改Truncated incorrect DOUBLE value:
  5. 深入浅出的理解框架(Struts2、Hibernate、Spring)与 MVC 设计模式
  6. eclipse中对单独JS文件取消报错的处理
  7. 单词查找树(信息学奥赛一本通-T1337)
  8. x86汇编语言——处理器架构
  9. UVa 120 - Stacks of Flapjacks
  10. 2021年中国一次性医用传感器市场趋势报告、技术动态创新及2027年市场预测
  11. python面向对象思路_Python面向对象三要素-继承(Inheritance)
  12. 十二客推出新版邮箱批量注册
  13. XQuery FLWOR 表达式
  14. 与代码无关的网络安全
  15. 10-25 查询选修张老师讲授所有课程的学生
  16. js设计程序实现摄氏度和华氏度转换
  17. Alpha版本冲刺(七)
  18. java contions_十一届蓝桥模拟赛 元辅音字母 JAVA
  19. SSRPanel 后端配置对接教程
  20. 我做淘宝7年的工作经验总结

热门文章

  1. 最小代价(区间dp)(ybtoj)
  2. 多重背包的二进制优化(ybtoj-宝物筛选)
  3. CF702F-T-Shirts【FhqTreap】
  4. P4549-[模板]裴蜀定理
  5. P1967,ssl2267-货车运输【树上倍增LCA,最小生成树变形kruskal】
  6. 【2018.5.12】模拟赛之四-ssl2416 条形图【高精度,dp】
  7. 写一个http服务器
  8. 跟我学 Java 8 新特性之 Stream 流(七)流与迭代器,流系列大结局
  9. MySQL datediff()函数
  10. 使用ADO.NET查询和操作数据