热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

POJO类中布尔类型的变量都不要加is前缀详解

前言对应阿里巴巴开发手册第一章的命名风格的第八条。【强制】POJO类中布尔类型的变量都不要加is前缀,否则部分框架解析会引起序列化错误。反例:定义

前言

对应阿里巴巴开发手册第一章的命名风格的第八条。

【强制】 POJO类中布尔类型的变量都不要加is前缀,否则部分框架解析会引起序列化错误。

反例:定义为基本数据类型Boolean isDeleted; 的属性,它的方法名称也是 isDeleted(),
RPC框架在反向解析的时候,“误以为”对应的属性名称是deleted,导致属性获取不到抛出异常。

我对这个反例感觉有点怪怪的,基本数据类型Boolean?而且Boolean生成的getter方法是getXxx(),boolean生成的getter方法是isXxx(),疑惑,不知道是不是手册写错了还是我错了。

我认为这条很重要很重要。前后端传递数据时,就有可能因为布尔类型变量的命名,导致前后端传送数据时传递失败。 因为这个布尔类型的数据解析不一致,在后端该命名是isXxx,而在前端发送表单是传送的数据是isXxx,但是响应时却是Xxx,说明传输到后端时会解析成Xxx(像spring会根据getter和setter来解析POJO类,而当我创建的是isXxx,那么自动生成的getter方法会是isXxx方法,然后解析时会去掉is,所以变成了Xxx),而后端原本是isXxx,导致找不到该属性,所以值也传递不了。
在这里插入图片描述
在这里插入图片描述
所以这是一个致命问题,但是可以避免啊。


详解

在Java中布尔类型有基本数据类型和包装类,所以有四种方式来定义一个布尔类型的变量:

boolean isLive;
boolean live;
Boolean isLive;
Boolean live;

首先来总结上面的区别:


  1. 四种中有两种是boolean,而另外两种是Boolean。可不要傻乎乎的说这两个类型是一样的,正确的是它们是有区别的。
  2. 四种中有两种变量名是以is开头,另外两种没有。


先根据手册,可得布尔类型的变量命名不使用以is开头的。为什么?

来看看下面代码:

// boolean isLive
class People {private boolean isLive;public boolean isLive() {return isLive;}public void setLive(boolean live) {isLive = live;}
}// boolean live
class People {private boolean live;public boolean isLive() {return live;}public void setLive(boolean live) {this.live = live;}
}// Boolean isLive
class People {private Boolean isLive;public Boolean getLive() {return isLive;}public void setLive(Boolean live) {isLive = live;}
}// Boolean live
class People {private Boolean live;public Boolean getLive() {return live;}public void setLive(Boolean live) {this.live = live;}
}

噫,通过代码和idea自动生成的getter和setter方法,可得到:


  1. boolean类型的getter方法是isXxx()的形式。
  2. Boolean类型的getter方法是getXxx()的形式。
  3. 这两个类型的setter方法相同。

其实isXxx()方法和getXxx()都是一样的,都是会解析成Xxx,但是只有布尔类型才有isXxx()方法,所以两种可以互换。比如把Boolean类型的getXxx改成isXxx,或者把boolean类型的isXxx改成getXxx。

一般情况下,其实布尔类型使用isXxx或不使用没有区别,但是在序列化时才可以看到区别,特别是现在开发中,都是序列化或者自动序列化的,所以可能因为命名的问题导致错误。

现在就先解决使用isXxx和使用Xxx的问题,我们拿常用的JSON序列化举例,比如看看fastJson、jackson和Gson之间有何区别:(还好参考了别人的博文不然我自己举的例子只是fastJson)

// boolean isLive
class People1 implements Serializable {private boolean isLive;public boolean isLive() {return isLive;}public void setLive(boolean live) {isLive = live;}
}
// boolean live
class People2 implements Serializable {private boolean live;public boolean isLive() {return live;}public void setLive(boolean live) {this.live = live;}
}
public class PeopleTest {public static void main(String[] args) throws JsonProcessingException {ObjectMapper jackson = new ObjectMapper();Gson gson = new Gson();People1 people1 = new People1();people1.setLive(true);System.out.println("boolean isLive");System.out.println("fastJson:" + JSON.toJSONString(people1));System.out.println("jackson:" + jackson.writeValueAsString(people1));System.out.println("Gson:" + gson.toJson(people1));People2 people2 = new People2();people2.setLive(true);System.out.println("boolean live");System.out.println("fastJson:" + JSON.toJSONString(people2));System.out.println("jackson:" + jackson.writeValueAsString(people2));System.out.println("Gson:" + gson.toJson(people2));}
}

去maven仓库访问速度有时太慢了,所以下面提供了fastJson、jackson和Gson的仓库链接,自己拿去耍耍。

<dependency><groupId>com.alibabagroupId><artifactId>fastjsonartifactId><version>1.2.58version>dependency><dependency><groupId>com.fasterxml.jackson.coregroupId><artifactId>jackson-coreartifactId><version>2.9.8version>dependency><dependency><groupId>com.google.code.gsongroupId><artifactId>gsonartifactId><version>2.8.5version>dependency>

运行结果&#xff1a;

boolean isLive
fastJson&#xff1a;{"live":true}
jackson&#xff1a;{"live":true}
Gson&#xff1a;{"isLive":true}
boolean live
fastJson&#xff1a;{"live":true}
jackson&#xff1a;{"live":true}
Gson&#xff1a;{"live":true}

根据这些结果就可以得出&#xff1a;


  1. 使用布尔类型的变量命名为isXxx&#xff0c;fastJson和jackson在把对象序列化成json字符串时&#xff0c;是通过反射遍历出该类中的所有getter方法(或isXxx方法)&#xff0c;得到isLive()方法&#xff0c;然后根据JavaBean规范&#xff0c;它会认为这是live属性&#xff0c;然后序列化成json。
  2. 而Gson并不是这么做的&#xff0c;他是通过反射遍历该类中的所有属性&#xff0c;并把其值序列化成json。

可得由于不同的序列化工具&#xff0c;在进行序列化的时候使用到的策略是不一样的&#xff0c;所以&#xff0c;对于同一个类的同一个对象的序列化结果可能是不同的。

现在&#xff0c;不同的序列化框架得到的json内容并不相同&#xff0c;如果对于同一个对象&#xff0c;我使用fastjson进行序列化&#xff0c;再使用Gson反序列化会发生什么&#xff1f;

// boolean isLive
class People1 implements Serializable {private boolean isLive;public boolean isLive() {return isLive;}public void setLive(boolean live) {isLive &#61; live;}&#64;Overridepublic String toString() {return "People1{" &#43;"isLive&#61;" &#43; isLive &#43;&#39;}&#39;;}
}
// boolean live
class People2 implements Serializable {private boolean live;public boolean isLive() {return live;}public void setLive(boolean live) {this.live &#61; live;}&#64;Overridepublic String toString() {return "People2{" &#43;"live&#61;" &#43; live &#43;&#39;}&#39;;}
}
public class PeopleTest {public static void main(String[] args) throws JsonProcessingException {Gson gson &#61; new Gson();People1 people1 &#61; new People1();people1.setLive(true);People2 people2 &#61; new People2();people2.setLive(true);System.out.println("boolean isLive::" &#43; gson.fromJson(JSON.toJSONString(people1), People1.class));System.out.println("boolean live::" &#43; gson.fromJson(JSON.toJSONString(people2), People2.class));}
}

结果&#xff1a;

在这里插入图片描述

会发现&#xff0c;在People1类中已经把isLive设置成true&#xff0c;现在怎么变成false&#xff1f;

原因是因为JSON框架通过扫描所有的getter后发现有一个isLive方法&#xff0c;然后根据JavaBeans的规范&#xff0c;解析出变量名为live&#xff0c;把model对象序列化城字符串后内容为{“live”: true}。
然后根据{“live”: true}这个json串&#xff0c;Gson框架在通过解析后&#xff0c;通过反射寻找People类中的live属性&#xff0c;但是People类中只有isLive属性&#xff0c;找不到对应的属性&#xff0c;所以&#xff0c;最终反序列化后的People类的对象中&#xff0c;isLive则会使用默认值false。

所以这样会导致前台的布尔类型的数据传送不到后端&#xff0c;因为前端传过来的会把命名为isXxx的变量解析成Xxx&#xff0c;而我们POJO类中的布尔类型的属性是isXXX&#xff0c;导致在POJO类中找不到Xxx&#xff0c;所以就导致传递值失败。我们必须避免这样的致命问题。

最终&#xff0c;解释了&#xff1a;POJO类中布尔类型的变量都不要加is前缀



下面再来解释&#xff0c;是使用Boolean类型还是使用boolean。这里可以看开发手册 1.4 OOP规约的第11条


所有的POJO类属性必须使用包装数据类型。
RPC方法的返回值和参数必须使用包装数据类型。
所有的局部变量使用基本数据类型。


为什么&#xff1f;

很简单&#xff0c;看这两种类型的区别&#xff0c;boolean类型的默认值为false&#xff1b;而Boolean类型的默认值是null。

举个栗子&#xff1a;在学校考试的时候&#xff0c;我们要录入学生成绩&#xff0c;假设使用Integer类型来录入&#xff0c;那么学生要是没有来考试&#xff0c;那么该成绩就是null&#xff0c;学生考0分&#xff0c;说明学生有来考试&#xff0c;但是考了0分&#xff1b;使用int类型来录入&#xff0c;注意int类型默认值为0&#xff0c;那么成绩如果是0&#xff0c;我们不能确定该学生是没有来考试的呢&#xff1f;还是有来考试但考了0分。

参考&#xff1a;
为什么阿里巴巴开发手册强制规定POJO 类中布尔类型的变量&#xff0c;都不要加 is 前缀


推荐阅读
  • 本文介绍了如何在iOS平台上使用GLSL着色器将YV12格式的视频帧数据转换为RGB格式,并展示了转换后的图像效果。通过详细的技术实现步骤和代码示例,读者可以轻松掌握这一过程,适用于需要进行视频处理的应用开发。 ... [详细]
  • Squaretest:自动生成功能测试代码的高效插件
    本文将介绍一款名为Squaretest的高效插件,该工具能够自动生成功能测试代码。使用这款插件的主要原因是公司近期加强了代码质量的管控,对各项目进行了严格的单元测试评估。Squaretest不仅提高了测试代码的生成效率,还显著提升了代码的质量和可靠性。 ... [详细]
  • 在Kohana 3框架中,实现最优的即时消息显示方法是许多开发者关注的问题。本文将探讨如何高效、优雅地展示flash消息,包括最佳实践和技术细节,以提升用户体验和代码可维护性。 ... [详细]
  • 本文探讨了 Java 中 Pair 类的历史与现状。虽然 Java 标准库中没有内置的 Pair 类,但社区和第三方库提供了多种实现方式,如 Apache Commons 的 Pair 类和 JavaFX 的 javafx.util.Pair 类。这些实现为需要处理成对数据的开发者提供了便利。此外,文章还讨论了为何标准库未包含 Pair 类的原因,以及在现代 Java 开发中使用 Pair 类的最佳实践。 ... [详细]
  • 在Android 4.4系统中,通过使用 `Intent` 对象并设置动作 `ACTION_GET_CONTENT` 或 `ACTION_OPEN_DOCUMENT`,可以从相册中选择图片并获取其路径。具体实现时,需要为 `Intent` 添加相应的类别,并处理返回的 Uri 以提取图片的文件路径。此方法适用于需要从用户相册中选择图片的应用场景,能够确保兼容性和用户体验。 ... [详细]
  • 本文介绍了UUID(通用唯一标识符)的概念及其在JavaScript中生成Java兼容UUID的代码实现与优化技巧。UUID是一个128位的唯一标识符,广泛应用于分布式系统中以确保唯一性。文章详细探讨了如何利用JavaScript生成符合Java标准的UUID,并提供了多种优化方法,以提高生成效率和兼容性。 ... [详细]
  • SQLite数据库CRUD操作实例分析与应用
    本文通过分析和实例演示了SQLite数据库中的CRUD(创建、读取、更新和删除)操作,详细介绍了如何在Java环境中使用Person实体类进行数据库操作。文章首先阐述了SQLite数据库的基本概念及其在移动应用开发中的重要性,然后通过具体的代码示例,逐步展示了如何实现对Person实体类的增删改查功能。此外,还讨论了常见错误及其解决方法,为开发者提供了实用的参考和指导。 ... [详细]
  • 从零起步:使用IntelliJ IDEA搭建Spring Boot应用的详细指南
    从零起步:使用IntelliJ IDEA搭建Spring Boot应用的详细指南 ... [详细]
  • 探讨 `org.openide.windows.TopComponent.componentOpened()` 方法的应用及其代码实例分析 ... [详细]
  • ButterKnife 是一款用于 Android 开发的注解库,主要用于简化视图和事件绑定。本文详细介绍了 ButterKnife 的基础用法,包括如何通过注解实现字段和方法的绑定,以及在实际项目中的应用示例。此外,文章还提到了截至 2016 年 4 月 29 日,ButterKnife 的最新版本为 8.0.1,为开发者提供了最新的功能和性能优化。 ... [详细]
  • 本文详细探讨了在ASP.NET环境中通过加密数据库连接字符串来提升数据安全性的方法。加密技术不仅能够有效防止敏感信息泄露,还能增强应用程序的整体安全性。文中介绍了多种加密手段及其实施步骤,帮助开发者在日常开发过程中更好地保护数据库连接信息,确保数据传输的安全可靠。 ... [详细]
  • 尽管我们尽最大努力,任何软件开发过程中都难免会出现缺陷。为了更有效地提升对支持部门的协助与支撑,本文探讨了多种策略和最佳实践,旨在通过改进沟通、增强培训和支持流程来减少这些缺陷的影响,并提高整体服务质量和客户满意度。 ... [详细]
  • 本文探讨了 Kafka 集群的高效部署与优化策略。首先介绍了 Kafka 的下载与安装步骤,包括从官方网站获取最新版本的压缩包并进行解压。随后详细讨论了集群配置的最佳实践,涵盖节点选择、网络优化和性能调优等方面,旨在提升系统的稳定性和处理能力。此外,还提供了常见的故障排查方法和监控方案,帮助运维人员更好地管理和维护 Kafka 集群。 ... [详细]
  • 本文探讨了资源访问的学习路径与方法,旨在帮助学习者更高效地获取和利用各类资源。通过分析不同资源的特点和应用场景,提出了多种实用的学习策略和技术手段,为学习者提供了系统的指导和建议。 ... [详细]
  • Spring Boot 实战(一):基础的CRUD操作详解
    在《Spring Boot 实战(一)》中,详细介绍了基础的CRUD操作,涵盖创建、读取、更新和删除等核心功能,适合初学者快速掌握Spring Boot框架的应用开发技巧。 ... [详细]
author-avatar
isonlyk
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有