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

java布尔字段用is前缀_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,导致找不到该属性,所以值也传递不了。

9493ac182adee675b56fed74633147b9.png

c4f89cf994d48fb069b64782cd57bb37.png

所以这是一个致命问题,但是可以避免啊。

详解

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

boolean isLive;

boolean live;

Boolean isLive;

Boolean live;

首先来总结上面的区别:

四种中有两种是boolean,而另外两种是Boolean。可不要傻乎乎的说这两个类型是一样的,正确的是它们是有区别的。

四种中有两种变量名是以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方法,可得到:

boolean类型的getter方法是isXxx()的形式。

Boolean类型的getter方法是getXxx()的形式。

这两个类型的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的仓库链接,自己拿去耍耍。

com.alibaba

fastjson

1.2.58

com.fasterxml.jackson.core

jackson-core

2.9.8

com.google.code.gson

gson

2.8.5

运行结果:

boolean isLive

fastJson:{"live":true}

jackson:{"live":true}

Gson:{"isLive":true}

boolean live

fastJson:{"live":true}

jackson:{"live":true}

Gson:{"live":true}

根据这些结果就可以得出:

使用布尔类型的变量命名为isXxx,fastJson和jackson在把对象序列化成json字符串时,是通过反射遍历出该类中的所有getter方法(或isXxx方法),得到isLive()方法,然后根据JavaBean规范,它会认为这是live属性,然后序列化成json。

而Gson并不是这么做的,他是通过反射遍历该类中的所有属性,并把其值序列化成json。

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

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

// boolean isLive

class People1 implements Serializable {

private boolean isLive;

public boolean isLive() {

return isLive;

}

public void setLive(boolean live) {

isLive = live;

}

@Override

public String toString() {

return "People1{" +

"isLive=" + isLive +

'}';

}

}

// boolean live

class People2 implements Serializable {

private boolean live;

public boolean isLive() {

return live;

}

public void setLive(boolean live) {

this.live = live;

}

@Override

public String toString() {

return "People2{" +

"live=" + live +

'}';

}

}

public class PeopleTest {

public static void main(String[] args) throws JsonProcessingException {

Gson gson = new Gson();

People1 people1 = new People1();

people1.setLive(true);

People2 people2 = new People2();

people2.setLive(true);

System.out.println("boolean isLive::" + gson.fromJson(JSON.toJSONString(people1), People1.class));

System.out.println("boolean live::" + gson.fromJson(JSON.toJSONString(people2), People2.class));

}

}

结果:

2d73f168254b0307f360f0c9229e5d98.png

会发现,在People1类中已经把isLive设置成true,现在怎么变成false?

原因是因为JSON框架通过扫描所有的getter后发现有一个isLive方法,然后根据JavaBeans的规范,解析出变量名为live,把model对象序列化城字符串后内容为{“live”: true}。

然后根据{“live”: true}这个json串,Gson框架在通过解析后,通过反射寻找People类中的live属性,但是People类中只有isLive属性,找不到对应的属性,所以,最终反序列化后的People类的对象中,isLive则会使用默认值false。

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

最终,解释了:POJO类中布尔类型的变量都不要加is前缀。

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

所有的POJO类属性必须使用包装数据类型。

RPC方法的返回值和参数必须使用包装数据类型。

所有的局部变量使用基本数据类型。

为什么?

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

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



推荐阅读
  • 本文总结了JavaScript的核心知识点和实用技巧,涵盖了变量声明、DOM操作、事件处理等重要方面。例如,通过`event.srcElement`获取触发事件的元素,并使用`alert`显示其HTML结构;利用`innerText`和`innerHTML`属性分别设置和获取文本内容及HTML内容。此外,还介绍了如何在表单中动态生成和操作``元素,以便更好地处理用户输入。这些技巧对于提升前端开发效率和代码质量具有重要意义。 ... [详细]
  • JavaScript XML操作实用工具类:XmlUtilsJS技巧与应用 ... [详细]
  • Java学习第10天:深入理解Map接口及其应用 ... [详细]
  • 本文探讨了如何利用 jQuery 的 JSONP 技术实现跨域调用外部 Web 服务。通过详细解析 JSONP 的工作原理及其在 jQuery 中的应用,本文提供了实用的代码示例和最佳实践,帮助开发者解决跨域请求中的常见问题。 ... [详细]
  • 在使用SSH框架进行项目开发时,经常会遇到一些常见的问题。例如,在Spring配置文件中配置AOP事务声明后,进行单元测试时可能会出现“No Hibernate Session bound to thread”的错误。本文将详细探讨这一问题的原因,并提供有效的解决方案,帮助开发者顺利解决此类问题。 ... [详细]
  • 探索偶数次幂二项式系数的求和方法及其数学意义 ... [详细]
  • 本文详细探讨了Zebra路由软件中的线程机制及其实际应用。通过对Zebra线程模型的深入分析,揭示了其在高效处理网络路由任务中的关键作用。文章还介绍了线程同步与通信机制,以及如何通过优化线程管理提升系统性能。此外,结合具体应用场景,展示了Zebra线程机制在复杂网络环境下的优势和灵活性。 ... [详细]
  • Objective-C 中的委托模式详解与应用 ... [详细]
  • 本文介绍了如何在iOS平台上使用GLSL着色器将YV12格式的视频帧数据转换为RGB格式,并展示了转换后的图像效果。通过详细的技术实现步骤和代码示例,读者可以轻松掌握这一过程,适用于需要进行视频处理的应用开发。 ... [详细]
  • 本文深入解析了Java 8并发编程中的`AtomicInteger`类,详细探讨了其源码实现和应用场景。`AtomicInteger`通过硬件级别的原子操作,确保了整型变量在多线程环境下的安全性和高效性,避免了传统加锁方式带来的性能开销。文章不仅剖析了`AtomicInteger`的内部机制,还结合实际案例展示了其在并发编程中的优势和使用技巧。 ... [详细]
  • DRF框架中Serializer反序列化验证机制详解:深入探讨Validators的应用与优化
    在DRF框架的反序列化验证机制中,除了基本的字段类型和长度校验外,还常常需要进行更为复杂的条件限制校验。通过引入`validators`模块,可以实现自定义校验逻辑,如唯一字段校验等。本文将详细探讨`validators`的使用方法及其优化策略,帮助开发者更好地理解和应用这一重要功能。 ... [详细]
  • 本文探讨了 Java 中 Pair 类的历史与现状。虽然 Java 标准库中没有内置的 Pair 类,但社区和第三方库提供了多种实现方式,如 Apache Commons 的 Pair 类和 JavaFX 的 javafx.util.Pair 类。这些实现为需要处理成对数据的开发者提供了便利。此外,文章还讨论了为何标准库未包含 Pair 类的原因,以及在现代 Java 开发中使用 Pair 类的最佳实践。 ... [详细]
  • 在前文探讨了Spring如何为特定的bean选择合适的通知器后,本文将进一步深入分析Spring AOP框架中代理对象的生成机制。具体而言,我们将详细解析如何通过代理技术将通知器(Advisor)中包含的通知(Advice)应用到目标bean上,以实现切面编程的核心功能。 ... [详细]
  • 通过使用CIFAR-10数据集,本文详细介绍了如何快速掌握Mixup数据增强技术,并展示了该方法在图像分类任务中的显著效果。实验结果表明,Mixup能够有效提高模型的泛化能力和分类精度,为图像识别领域的研究提供了有价值的参考。 ... [详细]
  • 在C#中开发MP3播放器时,我正在考虑如何高效存储元数据以便快速检索。选择合适的数据结构,如字典或数组,对于优化性能至关重要。字典能够提供快速的键值对查找,而数组则在连续存储和遍历方面表现优异。根据具体需求,合理选择数据结构将显著提升应用的响应速度和用户体验。 ... [详细]
author-avatar
唯依爱时尚服饰
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有