Java避坑指南:POJO类属性建议一律为包装类型,而且不要设置任何属性默认值

简介: Java避坑指南:POJO类属性建议一律为包装类型,而且不要设置任何属性默认值

首发地址:https://mp.weixin.qq.com/s?__biz=Mzg4MzcwMTk0Mw==&mid=2247484226&idx=1&sn=3f31526d257537fb9e3db84cc9bb0695&chksm=cf42229af835ab8c7dbb1b4f980e65bee4f49623eb5fdbf01b6e91782e3dcf0bfb5bfdc4d604&token=216307076&lang=zh_CN#rd

**【建议】POJO类属性建议一律为包装类型,而且不要设置任何属性默认值
**

如果在开发中,我们对POJO类属性设置了默认值(包装类型显示设置默认值,基本类型编译期推导默认值),难免会遇到一些"坑"。

坑:反序列化可能导致默认值可以被null覆盖

以jackson为例:

package com.renzhikeji.demo;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.apache.commons.lang3.builder.ToStringBuilder;


/**
 * @author 认知科技技术团队
 * 微信公众号:认知科技技术团队
 */
public class Demo {
    public static void main(String[] args) throws JsonProcessingException {

        ObjectMapper mapper = new ObjectMapper();
        String json = "{\"id\":null,\"name\":null,\"address\":\"\"}";
        POJO value = mapper.readValue(json, POJO.class);
        System.out.println(value);
    }

    public static class POJO {
        private Long id = 0L;
        private String name;
        private String address = "";

        public Long getId() {
            return id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public String getAddress() {
            return address;
        }

        public void setAddress(String address) {
            this.address = address;
        }

        @Override
        public String toString() {
            return new ToStringBuilder(this)
                    .append("id", id)
                    .append("name", name)
                    .append("address", address)
                    .toString();
        }
    }
}

POJO定义中,id默认值为0,当我们的反序列化字符串为:
{"id":null,"name":null,"address":""}

json字符串中id设置为null的时候:结果反序列化后,id的默认值就没有了。

https://mmbiz.qpic.cn/mmbiz_png/K5AWcTRtu4Whs4g7BeelajfH0buUGxWhhq7GTf71YHla0s1MswFRt6LIeXExEHfS2TM0j1xm2zV3icc6KVcMqPw/640?wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1image.png

坑:POJO与DTO之间转换,一个为原生类型,一个是对应的包装类型,使用类似BeanUtils.copyProperties的工具复制对象可能会抛出异常

以org.springframework.beans.BeanUtils#copyProperties(java.lang.Object, java.lang.Object)为例(spring-beans-5.3.21):


package com.renzhikeji.demo;

import com.fasterxml.jackson.core.JsonProcessingException;
import org.apache.commons.lang3.builder.ToStringBuilder;
import org.springframework.beans.BeanUtils;


/**
 * @author 认知科技技术团队
 * 微信公众号:认知科技技术团队
 */
public class Demo {
    public static void main(String[] args) throws JsonProcessingException {
        POJO pojo = new POJO();
        pojo.setId(null);

        DTO dto = new DTO();
        BeanUtils.copyProperties(pojo, dto);
        System.out.println(dto);
    }

    public static class POJO {
        private Long id = 0L;
        private String name;
        private String address = "";

        public Long getId() {
            return id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public String getAddress() {
            return address;
        }

        public void setAddress(String address) {
            this.address = address;
        }

        @Override
        public String toString() {
            return new ToStringBuilder(this).append("id", id).append("name", name).append("address", address).toString();
        }
    }


    public static class DTO {
        private long id = 0L;
        private String name;
        private String address = "";

        public long getId() {
            return id;
        }

        public void setId(long id) {
            this.id = id;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public String getAddress() {
            return address;
        }

        public void setAddress(String address) {
            this.address = address;
        }

        @Override
        public String toString() {
            return new ToStringBuilder(this).append("id", id).append("name", name).append("address", address).toString();
        }
    }
}

主要区别是:同一个属性名,但是一个是原生类型,一个是对应的包装类型,原生类型的本来意愿是不传值,就有个默认值,可惜,事与愿违:

https://mmbiz.qpic.cn/mmbiz_png/K5AWcTRtu4Whs4g7BeelajfH0buUGxWhaibWbszmDtbdB3giaibwWWodcOZ3fXMzicnZugAxHcketa4kqA98RLHgicg/640?wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1image.png

**坑:MyBatis的动态sql中,可能遇到
**

MyBatis动态sql中,如果遇到如下所示类似的动态sql,title或author没有显示设置,POJO中都设置了默认值,动态sql就失去了意义,业务逻辑就错了。
<select id="findActiveBlogLike"

 resultType="Blog">

SELECT * FROM BLOG WHERE state = ‘ACTIVE’

AND title like #{title}


AND author_name like #{author.name}


**坑:API接口或RPC接口返回默认值,造成业务逻辑bug
**
拿部门同事的计费系统的一个扣费异常为例,扣费时需要通过 RPC 请求计费系统得到一个费率值,预期该接口的返回值中会包含一个浮点型的费率字段。
当我们取到这个值得时候就使用公式:金额*费率=费用 进行计算,计算结果进行划扣。
如果RPC返回该费率值时可能由于bug或其他业务原因没有设置,拿到默认值0.0就进行计算,不会进行扣费,这种扣费为0的异常情况无法被感知(当然你可以把这种业务情况发报警),但是如果不用原生类型或不设置包装类型默认值,以null参与计算直接抛出异常报警,是不是更容易发现。

小结

POJO类属性建议一律为包装类型,而且不要设置任何属性默认值,以上的坑只是冰山一角。

相关文章
|
4月前
|
存储 Java 索引
用Java语言实现一个自定义的ArrayList类
自定义MyArrayList类模拟Java ArrayList核心功能,支持泛型、动态扩容(1.5倍)、增删改查及越界检查,底层用Object数组实现,适合学习动态数组原理。
174 4
|
4月前
|
IDE JavaScript Java
在Java 11中,如何处理被弃用的类或接口?
在Java 11中,如何处理被弃用的类或接口?
263 5
|
4月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
249 1
|
4月前
|
Java Go 开发工具
【Java】(8)正则表达式的使用与常用类分享
正则表达式定义了字符串的模式。正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别。
309 1
|
4月前
|
存储 Java 程序员
【Java】(6)全方面带你了解Java里的日期与时间内容,介绍 Calendar、GregorianCalendar、Date类
java.util 包提供了 Date 类来封装当前的日期和时间。Date 类提供两个构造函数来实例化 Date 对象。第一个构造函数使用当前日期和时间来初始化对象。Date( )第二个构造函数接收一个参数,该参数是从1970年1月1日起的毫秒数。
240 0
|
4月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
266 1
|
4月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
345 5
|
5月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
222 0
|
5月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
412 16