Java对象空指针校验工具类

简介: 优雅的解决空指针错误

Java对象空指针校验工具类

📔 千寻简笔记介绍

千寻简笔记已开源,Gitee与GitHub搜索chihiro-notes,包含笔记源文件.md,以及PDF版本方便阅读,且是用了精美主题,阅读体验更佳,如果文章对你有帮助请帮我点一个Star

简介

对象空指针是指一个指针变量指向了内存中的空地址,也就是没有指向任何有效对象的地址。在许多编程语言中,空指针通常用特殊的值(例如NULL、nil、None等)表示。

当你使用一个空指针来访问对象的成员或调用对象的方法时,会导致空指针异常(Null Pointer Exception)或类似的错误。这是因为空指针并没有指向有效的对象,无法执行相应的操作。

空指针错误通常是由以下几种情况引起的:

  1. 未初始化指针:当你声明一个指针变量但没有给它赋予有效的地址时,它的值就是空指针。如果你在使用该指针之前没有对其进行初始化,就会导致空指针异常。
  2. 对象释放或销毁:如果你在释放或销毁一个对象后,仍然使用指向该对象的指针,就会得到空指针异常。这通常发生在动态内存分配的情况下,例如使用new关键字创建的对象。
  3. 对象不存在:当你试图访问一个不存在的对象或已经被销毁的对象时,指向该对象的指针就会成为空指针。这可能是由于对象在之前的代码中未被正确创建或意外地被销毁。

为了避免空指针错误,你可以在使用指针之前进行有效性检查,或者使用安全的编程技术,如空指针检查、异常处理等。

工具类

1 基于若依改动

1.1 调用示例

    /**
     * 根据id查询文章列表
     * 
     * @param article 文章
     * @return 文章
     */
    @Override
    public Article getArticleById(Long id)
    {
   
        Article article = baseMapper.selectArticleById(id);
        AssertUtils.isNotEmpty(article,"抛出异常");
        return article;
    }

AssertUtil:校验工具类

package com.torinosrc.common.utils.star;

import cn.hutool.core.util.ObjectUtil;

import com.torinosrc.common.exception.ServiceException;

import java.text.MessageFormat;
import java.util.Objects;

/**
 * 校验工具类
 */
public class AssertUtil {
   

    //如果不是true,则抛异常
    public static void isTrue(boolean expression, String msg) {
   
        if (!expression) {
   
            throwException(msg);
        }
    }

    public static void isTrue(boolean expression, ErrorEnum errorEnum, Object... args) {
   
        if (!expression) {
   
            throwException(errorEnum, args);
        }
    }

    //如果是true,则抛异常
    public static void isFalse(boolean expression, String msg) {
   
        if (expression) {
   
            throwException(msg);
        }
    }

    //如果是true,则抛异常
    public static void isFalse(boolean expression, ErrorEnum errorEnum, Object... args) {
   
        if (expression) {
   
            throwException(errorEnum, args);
        }
    }

    //如果不是非空对象,则抛异常
    public static void isNotEmpty(Object obj, String msg) {
   
        if (isEmpty(obj)) {
   
            throwException(msg);
        }
    }

    //如果不是非空对象,则抛异常
    public static void isNotEmpty(Object obj, ErrorEnum errorEnum, Object... args) {
   
        if (isEmpty(obj)) {
   
            throwException(errorEnum, args);
        }
    }

    //如果不是非空对象,则抛异常
    public static void isEmpty(Object obj, String msg) {
   
        if (!isEmpty(obj)) {
   
            throwException(msg);
        }
    }

    public static void equal(Object o1, Object o2, String msg) {
   
        if (!ObjectUtil.equal(o1, o2)) {
   
            throwException(msg);
        }
    }

    public static void notEqual(Object o1, Object o2, String msg) {
   
        if (ObjectUtil.equal(o1, o2)) {
   
            throwException(msg);
        }
    }

    private static boolean isEmpty(Object obj) {
   
        return ObjectUtil.isEmpty(obj);
    }

    private static void throwException(String msg) {
   
        throwException(null, msg);
    }

    private static void throwException(ErrorEnum errorEnum, Object... arg) {
   
        if (Objects.isNull(errorEnum)) {
   
            errorEnum = BusinessErrorEnum.BUSINESS_ERROR;
        }
        throw new ServiceException(errorEnum.getErrorCode(), MessageFormat.format(errorEnum.getErrorMsg(), arg));
    }
}

1.2 BusinessErrorEnum:业务校验异常码

package com.torinosrc.common.utils.star;


/**
 * Description: 业务校验异常码
 */
public enum BusinessErrorEnum implements ErrorEnum {
   
    //==================================common==================================
    BUSINESS_ERROR(1001, "{0}"),
    //==================================chat==================================
    SYSTEM_ERROR(1001, "系统出小差了,请稍后再试哦~~"),
    ;
    private Integer code;
    private String msg;

    BusinessErrorEnum(Integer code, String msg) {
   
        this.code = code;
        this.msg = msg;
    }

    @Override
    public Integer getErrorCode() {
   
        return code;
    }

    @Override
    public String getErrorMsg() {
   
        return msg;
    }

    public Integer getCode() {
   
        return code;
    }

    public BusinessErrorEnum setCode(Integer code) {
   
        this.code = code;
        return this;
    }

    public String getMsg() {
   
        return msg;
    }

    public BusinessErrorEnum setMsg(String msg) {
   
        this.msg = msg;
        return this;
    }
}

1.3 ServiceException:修改若依业务异常

package com.torinosrc.common.exception;

/**
 * 业务异常
 *
 * @author ruoyi
 */
public final class ServiceException extends RuntimeException
{
   
    private static final long serialVersionUID = 1L;

    /**
     * 错误码
     */
    private Integer code;

    /**
     * 错误提示
     */
    private String message;

    /**
     * 错误明细,内部调试错误
     *
     * 和 {@link CommonResult#getDetailMessage()} 一致的设计
     */
    private String detailMessage;

    /**
     * 空构造方法,避免反序列化问题
     */
    public ServiceException()
    {
   
    }

    public ServiceException(String message) {
   
        this.message = message;
    }

    public ServiceException(String message, Integer code) {
   
        this.message = message;
        this.code = code;
    }

    // 增加方法1
    public ServiceException(Integer errorCode, String errorMsg) {
   
        super(errorMsg);
        this.code = errorCode;
        this.message = errorMsg;
    }
     // 增加方法2
    public ServiceException(Integer errorCode, String errorMsg, Throwable cause) {
   
        super(errorMsg, cause);
        this.code = errorCode;
        this.message = errorMsg;
    }

    public String getDetailMessage() {
   
        return detailMessage;
    }

    public String getMessage() {
   
        return message;
    }

    public Integer getCode()
    {
   
        return code;
    }

    public ServiceException setMessage(String message)
    {
   
        this.message = message;
        return this;
    }

    public ServiceException setDetailMessage(String detailMessage)
    {
   
        this.detailMessage = detailMessage;
        return this;
    }
}

1.4 ErrorEnum:错误枚举类

package com.torinosrc.common.utils.star;

public interface ErrorEnum {
   

    Integer getErrorCode();

    String getErrorMsg();
}

2 其他项改动

下面提供一个空指针校验工具类:

2.1 调用示例:

    /**
     * 根据id查询文章列表
     * 
     * @param article 文章
     * @return 文章
     */
    @Override
    public Article getArticleById(Long id)
    {
        Article article = baseMapper.selectArticleById(id);
        AssertUtils.isNotEmpty(article,"抛出异常");
        return article;
    }

2.2 AssertUtil:校验工具类

import cn.hutool.core.util.ObjectUtil;


import java.text.MessageFormat;
import java.util.Objects;

/**
 * 校验工具类
 */
public class AssertUtil {
   

    //如果不是true,则抛异常
    public static void isTrue(boolean expression, String msg) {
   
        if (!expression) {
   
            throwException(msg);
        }
    }

    public static void isTrue(boolean expression, ErrorEnum errorEnum, Object... args) {
   
        if (!expression) {
   
            throwException(errorEnum, args);
        }
    }

    //如果是true,则抛异常
    public static void isFalse(boolean expression, String msg) {
   
        if (expression) {
   
            throwException(msg);
        }
    }

    //如果是true,则抛异常
    public static void isFalse(boolean expression, ErrorEnum errorEnum, Object... args) {
   
        if (expression) {
   
            throwException(errorEnum, args);
        }
    }

    //如果不是非空对象,则抛异常
    public static void isNotEmpty(Object obj, String msg) {
   
        if (isEmpty(obj)) {
   
            throwException(msg);
        }
    }

    //如果不是非空对象,则抛异常
    public static void isNotEmpty(Object obj, ErrorEnum errorEnum, Object... args) {
   
        if (isEmpty(obj)) {
   
            throwException(errorEnum, args);
        }
    }

    //如果不是非空对象,则抛异常
    public static void isEmpty(Object obj, String msg) {
   
        if (!isEmpty(obj)) {
   
            throwException(msg);
        }
    }

    public static void equal(Object o1, Object o2, String msg) {
   
        if (!ObjectUtil.equal(o1, o2)) {
   
            throwException(msg);
        }
    }

    public static void notEqual(Object o1, Object o2, String msg) {
   
        if (ObjectUtil.equal(o1, o2)) {
   
            throwException(msg);
        }
    }

    private static boolean isEmpty(Object obj) {
   
        return ObjectUtil.isEmpty(obj);
    }

    private static void throwException(String msg) {
   
        throwException(null, msg);
    }

    private static void throwException(ErrorEnum errorEnum, Object... arg) {
   
        if (Objects.isNull(errorEnum)) {
   
            errorEnum = BusinessErrorEnum.BUSINESS_ERROR;
        }
        throw new BusinessException(errorEnum.getErrorCode(), MessageFormat.format(errorEnum.getErrorMsg(), arg));
    }
}

2.3 BusinessErrorEnum:业务校验异常码

import lombok.AllArgsConstructor;
import lombok.Getter;

/**
 * Description: 业务校验异常码
 * Author: <a href="https://github.com/zongzibinbin">abin</a>
 * Date: 2023-03-26
 */
@AllArgsConstructor
@Getter
public enum BusinessErrorEnum implements ErrorEnum {
   
    //==================================common==================================
    BUSINESS_ERROR(1001, "{0}"),
    //==================================user==================================
    //==================================chat==================================
    SYSTEM_ERROR(1001, "系统出小差了,请稍后再试哦~~"),
    ;
    private Integer code;
    private String msg;

    @Override
    public Integer getErrorCode() {
   
        return code;
    }

    @Override
    public String getErrorMsg() {
   
        return msg;
    }
}

2.4 BusinessException:异常实体类

import lombok.Data;

@Data
public class BusinessException extends RuntimeException {
   
    private static final long serialVersionUID = 1L;

    /**
     *  错误码
     */
    protected Integer errorCode;

    /**
     *  错误信息
     */
    protected String errorMsg;

    public BusinessException() {
   
        super();
    }

    public BusinessException(String errorMsg) {
   
        super(errorMsg);
        this.errorMsg = errorMsg;
    }

    public BusinessException(Integer errorCode, String errorMsg) {
   
        super(errorMsg);
        this.errorCode = errorCode;
        this.errorMsg = errorMsg;
    }

    public BusinessException(Integer errorCode, String errorMsg, Throwable cause) {
   
        super(errorMsg, cause);
        this.errorCode = errorCode;
        this.errorMsg = errorMsg;
    }

    public BusinessException(ErrorEnum error) {
   
        super(error.getErrorMsg());
        this.errorCode = error.getErrorCode();
        this.errorMsg = error.getErrorMsg();
    }

    @Override
    public String getMessage() {
   
        return errorMsg;
    }

    @Override
    public synchronized Throwable fillInStackTrace() {
   
        return this;
    }
}

2.5 ErrorEnum:错误枚举类

public interface ErrorEnum {
   

    Integer getErrorCode();

    String getErrorMsg();
}
目录
相关文章
|
1月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
20天前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
46 4
|
2月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
48 17
|
1月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
2月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
29 6
|
1月前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
33 0
|
Java API 容器
Java8新特性之空指针异常的克星Optional类
Java8新特性之空指针异常的克星Optional类
124 0
|
13天前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
4天前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
4天前
|
安全 Java 开发者
Java中的多线程编程:从基础到实践
本文深入探讨了Java多线程编程的核心概念和实践技巧,旨在帮助读者理解多线程的工作原理,掌握线程的创建、管理和同步机制。通过具体示例和最佳实践,本文展示了如何在Java应用中有效地利用多线程技术,提高程序性能和响应速度。
27 1