java 判断一个对象是否为空对象

java 判断一个对象是否为空对象最近项目中遇到一个问题,在用户没填数据的时候,我们需要接收从前端传过来的对象为null,但是前端说他们一个一个判断特别麻烦,只能传个空对象过来,我第一个想法就是可以通过反射来判断对象是否为空。第一版:User.javapublicclassUser{privateStringusername;privateBooleanactive;priva…

大家好,又见面了,我是你们的朋友全栈君。

最近项目中遇到一个问题,在用户没填数据的时候,我们需要接收从前端传过来的对象为null,但是前端说他们一个一个判断特别麻烦,只能传个空对象过来,我第一个想法就是可以通过反射来判断对象是否为空。

第一版:

User.java

public class User { 
   
    private String username;

    private Boolean active;

    private Long id;
    // 省略get和set方法
}

ReflectUtil.java

public class ReflectUtil { 
   
    public static boolean isObjectNull(Object obj){ 
   
        if (obj != null) { 
   
            Class<?> objClass = obj.getClass();
            Method[] declaredMethods = objClass.getDeclaredMethods();
            if (declaredMethods.length > 0) { 
   
                int methodCount = 0; // get 方法数量
                int nullValueCount = 0; // 结果为空

                for (Method declaredMethod : declaredMethods) { 
   
                    String name = declaredMethod.getName();
                    if (name.startsWith("get") || name.startsWith("is")){ 
   
                        methodCount += 1;
                        try { 
   
                            Object invoke = declaredMethod.invoke(obj);
                            if (invoke == null) { 
   
                                nullValueCount += 1;
                            }
                        } catch (IllegalAccessException | InvocationTargetException e){ 
   
                            e.printStackTrace();
                        }
                    }
                }
                return methodCount == nullValueCount;
            }
        }
        return false;
    }
}

TestReflect.java

public class TestReflect { 
   
    public static void main(String[] args) { 
   
        User user = new User();
        System.out.println(ReflectUtil.isObjectNull(user));
    }
}

结果:

true

第一版 获取一个类的声明的方法,判断方法如果以get或者is开头就是get方法,然后通过反射调用改方法获取结果,再判断结果是否为空,如果结果为null的话就把nullValueCount+1,最后返回结果为空的值的数量和get方法数量比较的结果,如果两者数量相同则说明该对象为空,反之不为空。
第一版也可以判断一个对象是否为空,但前提是对象必须使用包装类,没有默认值的就不行了,当然你也可以根据类型和返回值结果来判断对象是否为空,但是如果想忽略某个属性不做判断,改起来就有点麻烦了。 后来想知道spring 的BeanUtils 是怎么实现属性复制的就看了一下,发现了新的方法,于是就有了第二版。

第二版:

/** * 判断对象是否为空, * @param obj * @param ignoreProperties 忽略的属性 * @return 如果get 方法的数量等于 属性为空的数量 返回true,否则false */
    public static boolean isNullObject(Object obj , String... ignoreProperties) throws IntrospectionException { 
   
        if (obj != null) { 
   
            Class<?> objClass = obj.getClass();
            BeanInfo beanInfo = Introspector.getBeanInfo(objClass);
            PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();

            List<String> ignoreList = (ignoreProperties != null ? Arrays.asList(ignoreProperties) : null);

            int count = 1; // 结果为空的属性数量 初始化为1 去除Object的getClass方法
            int propertyCount = propertyDescriptors.length; // 属性数量
            if (ignoreList != null){ 
   
                propertyCount -= ignoreList.size();
            }

            for (PropertyDescriptor propertyDescriptor : propertyDescriptors) { 
   
                Method readMethod = propertyDescriptor.getReadMethod();
                String name = propertyDescriptor.getName();
                if (readMethod != null && (ignoreList == null || !ignoreList.contains(name))) { 
   
                    Class<?> returnType = readMethod.getReturnType();
                    String typeName = returnType.getSimpleName();
                    Object invoke = null;
                    try { 
   
                        invoke = readMethod.invoke(obj);
                        if (invoke == null) { 
   
                            count+=1;
                        }else { 
   
                            switch (typeName) { 
   
                                case "String":
                                    if ("".equals(invoke.toString().trim())) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "Integer":
                                    if ((Integer) invoke <= 0) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "int":
                                    if ((int) invoke <= 0) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "double":
                                    if ((double) invoke <= 0.0d) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "Double":
                                    if ((Double) invoke <= 0.0D) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "float":
                                    if ((float) invoke <= 0.0f) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "Float":
                                    if ((Float) invoke <= 0.0F) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "Long":
                                    if ((Long) invoke <= 0L) { 
   
                                        count += 1;
                                    }
                                    break;
                                case "long":
                                    if ((long) invoke <= 0L) { 
   
                                        count += 1;
                                    }
                                    break;
                            }
                        }
                    } catch (IllegalAccessException | InvocationTargetException e) { 
   
                        e.printStackTrace();
                    }
                }
            }
            return propertyCount == count;
        }
        return true;
    }

第一版和第二版思想基本都是一样的,都是通过读方法去判断返回值是否为空,只不过第二版在第一版上加强了可以忽略属性这个功能。
通过spring 的beanutils发现PropertyDescriptor这个类,从名字看来是个属性描述器,描述属性相关的东西,通过属性描述器可以获取bean的属性名称,读写方法,使用起来还挺方便。
通过Introspector内省类的静态方法getBeanInfo(Class<?> beanClass)获取BeanInfo,然后通过BeanInfo对象的getPropertyDescriptors()就可以返回属性描述器。
由于没有太多研究就不多介绍了。
如果你还有其他方法判断一个对象是否为空请留言,谢谢

能力有限,水平一般,如有错误,请多指出。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/135219.html原文链接:https://javaforall.net

(0)
上一篇 2022年6月5日 下午4:00
下一篇 2022年6月5日 下午4:00


相关推荐

  • 排列组合公式的原理_有序排列组合公式

    排列组合公式的原理_有序排列组合公式绪论:加法原理、乘法原理#分类计数原理:做一件事,有n类办法,在第1类办法中有m1种不同的方法,在第2类办法中有m2种不同的方法,…,在第n类办法中有mn种不同的方法,那么完成这件事共有N=m1+m2+…+mn种不同的方法。分步计数原理:完成一件事,需要分成n个步骤,做第1步有m1种不同的方法,做第2步有m2种不同的方法,…,做第n步有mn种不同的方法,那么完成这件事共有N=m1×m2×⋯×mn种不同的方法。区别:分类计数原理是加法原理,不同的类加起来就是我要得到的总数;分步计数原理是乘法原理,是同一

    2022年10月21日
    5
  • Solr集成Tomcat

    Solr集成Tomcat

    2021年5月30日
    108
  • Python中isinstance用法

    Python中isinstance用法Isinstance 的用法是用来判断一个量是否是相应的类型 接受的参数一个是对象加一种类型 示范代码如下 a 1print isinstance a int print isinstance a float 返回 True nbsp nbsp False

    2026年3月26日
    3
  • netstat -an解读

    netstat -an解读显示网络状况 我们可以使用 netstat 来显示目前的联机状况 例如 netstat aActiveInter includingser ProtoRecv QSend Q nbsp nbsp LocalAddress nbsp nbsp nbsp state tcp4 nbsp nbsp nbsp nbsp nbsp nbsp nbsp 0 nbsp nbsp nbsp nbsp nbsp 20 nbsp nbsp www ssh nbsp nbsp nbsp nbsp

    2026年3月26日
    2
  • 计算机系统存储器 分类,存储器的分类

    计算机系统存储器 分类,存储器的分类存储器的分类存储器是计算机的重要组成部分之一 用来存储程序和数据 表征了计算机的 记忆 功能 1 按用途分类 内部存储器内部存储器又叫内存 是主存储器 用来存储当前正在使用的或经常使用的程序和数据 CPU 可以对他直接访问 存取速度较快 外部存储器外部存储器又叫外存 是辅助寄存器 外存的特点是容量大 所存的信息既可以修改也可以保存 存取速度较慢 要用专用的设备来管理 计算机工作时 一般由内存 ROM 中

    2026年3月17日
    2
  • 免ROOT卸载手机自带软件详细教程

    免ROOT卸载手机自带软件详细教程不用 ROOT 也能卸载手机自带软件 教程很详细 包你学会

    2026年3月20日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号