跳到主要内容
极客日志极客日志面向AI+效率的开发者社区
首页博客GitHub 精选镜像工具UI配色美学隐私政策关于联系
搜索内容 / 工具 / 仓库 / 镜像...⌘K搜索
注册
博客列表
Javajava

Java 注解与反射实战:自定义日志与参数校验注解实现

综述由AI生成Java 注解与反射实战通过自定义日志注解@Log 和参数校验注解@NotNull/@Range,演示了如何利用反射拦截方法调用并自动处理业务逻辑。包含注解定义、元注解说明、处理器实现及测试用例,涵盖方法开始时间记录、参数打印、耗时统计、异常捕获及参数合法性校验等功能。总结了指向范围、生命周期、属性精简等设计原则,以及反射解析优化技巧,对比了与 Spring 及 JSR-303 框架的差异,帮助开发者从使用者升级为设计者。

字节跳动发布于 2026/2/5更新于 2026/6/3481 浏览
Java 注解与反射实战:自定义日志与参数校验注解实现

文章配图

文章配图

前言:为什么需要自定义注解?

在日常开发中,我们经常遇到两类重复工作:

日志记录:每个重要方法都要写 "开始执行"、"参数是 xxx"、"执行结束" 的代码;参数校验:判断输入是否为 null、年龄是否在合理范围、手机号格式是否正确等。

这些工作机械且冗余,而注解 + 反射正是解决这类问题的 "银弹"—— 用注解标记需要处理的地方,用反射自动执行逻辑,实现 "一次定义,多处复用"。

本文将带你从零实现两个实用案例:

  1. 自定义日志注解@Log:自动记录方法调用细节;
  2. 自定义参数校验注解@NotNull、@Range:自动校验方法参数合法性。

全程实战,代码可直接运行,搭配图解帮你吃透底层逻辑。

案例一:自定义日志注解@Log—— 自动记录方法调用轨迹

需求分析

我们需要一个注解,标记在方法上后,能自动完成:

记录方法开始执行的时间;打印方法参数(可选);记录方法执行耗时;打印返回结果(可选);捕获并记录方法抛出的异常。

步骤 1:定义@Log注解
import java.lang.annotation.*; // 只能标记在方法上 @Target(ElementType.METHOD) // 运行时保留,允许反射获取 @Retention(RetentionPolicy.RUNTIME) public @interface Log { // 操作描述(如"创建订单") String description() default ""; // 是否记录参数 boolean recordParams() default true; // 是否记录返回值 boolean recordResult() default true; // 是否记录异常 boolean recordException() default true; }

元注解说明:

@Target(ElementType.METHOD):限制注解仅用于方法(符合日志记录的场景);@Retention(RetentionPolicy.RUNTIME):必须保留到运行时,否则反射无法获取。

步骤 2:创建使用@Log注解的业务类
public class OrderService { // 标记日志:记录参数和返回值,描述为"创建订单" @Log(description = "创建订单", recordParams = true, recordResult = true) public String createOrder(String userId, double amount) { if (amount <= 0) { throw new IllegalArgumentException("金额必须大于 0"); } // 模拟业务耗时 try { Thread.sleep(100); } catch (InterruptedException e) {} return "ORDER_" + System.currentTimeMillis(); } // 标记日志:不记录返回值(无返回值),描述为"取消订单" @Log(description = "取消订单", recordParams = true, recordResult = false) public void cancelOrder(String orderId) { if (orderId == null || orderId.isEmpty()) { throw new IllegalArgumentException("订单 ID 不能为空"); } // 模拟业务耗时 try { Thread.sleep(50); } catch (InterruptedException e) {} }
}
步骤 3:实现注解解析器(核心逻辑)

通过反射拦截方法调用,解析@Log注解并执行日志记录逻辑:

import java.lang.reflect.Method;
import java.util.Arrays;

public class LogAnnotationProcessor {
    /**
     * 执行带日志注解的方法
     * @param target 目标对象
     * @param methodName 方法名
     * @param args 方法参数
     * @return 方法返回值
     */
    public static Object executeWithLog(Object target, String methodName, Object... args) {
        try {
            // 1. 获取方法对应的 Class 对象和参数类型
            Class<?>[] paramTypes = Arrays.stream(args)
                .map(arg -> arg == null ? Object.class : arg.getClass())
                .toArray(Class[]::new);
            Method method = target.getClass().getMethod(methodName, paramTypes);
            // 2. 检查方法是否有@Log 注解,无则直接执行
            if (!method.isAnnotationPresent(Log.class)) {
                return method.invoke(target, args);
            }
            // 3. 解析注解属性
            Log logAnnotation = method.getAnnotation(Log.class);
            String description = logAnnotation.description();
            boolean recordParams = logAnnotation.recordParams();
            boolean recordResult = logAnnotation.recordResult();
            boolean recordException = logAnnotation.recordException();
            // 4. 记录方法开始日志
            long startTime = System.currentTimeMillis();
            System.out.println("\n===== 【日志开始】" + (description.isEmpty() ? methodName : description) + " =====");
            if (recordParams) {
                System.out.println("参数:" + Arrays.toString(args));
            }
            // 5. 执行目标方法(捕获异常并记录)
            Object result;
            try {
                result = method.invoke(target, args);
            } catch (Exception e) {
                // 记录异常信息
                if (recordException) {
                    System.out.println("执行异常:" + e.getCause().getMessage());
                }
                throw e; // 继续抛出异常,不掩盖业务逻辑
            }
            // 6. 记录方法结束日志
            long endTime = System.currentTimeMillis();
            System.out.println("耗时:" + (endTime - startTime) + "ms");
            if (recordResult) {
                System.out.println("返回值:" + result);
            }
            System.out.println("===== 【日志结束】" + (description.isEmpty() ? methodName : description) + " =====");
            return result;
        } catch (Exception e) {
            // 处理反射或业务异常(实际项目中可转为自定义异常)
            throw new RuntimeException("方法执行失败:" + e.getMessage(), e);
        }
    }
}
步骤 4:测试日志注解效果
public class LogTest {
    public static void main(String[] args) {
        OrderService orderService = new OrderService();
        // 测试正常创建订单
        LogAnnotationProcessor.executeWithLog(orderService, "createOrder", "user_001", 99.9);
        // 测试取消订单
        LogAnnotationProcessor.executeWithLog(orderService, "cancelOrder", "ORDER_123456");
        // 测试异常场景(金额为负数)
        try {
            LogAnnotationProcessor.executeWithLog(orderService, "createOrder", "user_002", -10.0);
        } catch (Exception e) {
            // 此处仅捕获,不处理(异常已被日志记录)
        }
    }
}
执行结果
===== 【日志开始】创建订单 ===== 
参数:[user_001, 99.9] 
耗时:101ms 
返回值:ORDER_1698765432100 
===== 【日志结束】创建订单 ===== 
===== 【日志开始】取消订单 ===== 
参数:[ORDER_123456] 
耗时:50ms 
===== 【日志结束】取消订单 ===== 
===== 【日志开始】创建订单 ===== 
参数:[user_002, -10.0] 
执行异常:金额必须大于 0 
===== 【日志结束】创建订单 ===== 
日志注解执行流程图解

文章配图

(注:核心展示流程节点:调用方法→反射检查注解→记录开始日志→执行方法→记录结束日志)

案例二:自定义参数校验注解 —— 告别重复的 if-else

需求分析

我们需要两个注解:

@NotNull:标记参数不能为 null;@Range:标记数值参数必须在指定范围内(如年龄 1-120 岁)。

实现效果:方法调用时自动校验参数,不符合规则则抛出明确的异常信息。

步骤 1:定义校验注解
1.1 @NotNull注解
import java.lang.annotation.*;

@Target(ElementType.PARAMETER) // 仅用于方法参数
@Retention(RetentionPolicy.RUNTIME)
public @interface NotNull {
    // 校验失败的提示信息
    String message() default "参数不能为 null";
}
1.2 @Range注解
import java.lang.annotation.*;

@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface Range {
    long min() default 0; // 最小值(包含)
    long max() default Long.MAX_VALUE; // 最大值(包含)
    String message() default "参数不在指定范围内";
}
步骤 2:创建使用校验注解的业务类
public class UserService {
    // 新增用户:ID 不能为空,年龄必须 1-120 岁
    public void addUser(
        @NotNull(message = "用户 ID 不能为空") String userId,
        @Range(min = 1, max = 120, message = "年龄必须在 1-120 岁之间") int age) {
        System.out.println("新增用户成功:userId=" + userId + ", age=" + age);
    }
    // 更新积分:积分必须≥0
    public void updatePoints(
        @NotNull String userId,
        @Range(min = 0, message = "积分不能为负数") int points) {
        System.out.println("更新积分成功:userId=" + userId + ", points=" + points);
    }
}
步骤 3:实现参数校验器(核心逻辑)

通过反射获取方法参数上的注解,逐个校验参数是否符合注解规则:

import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.util.Arrays;

public class ParamValidator {
    /**
     * 校验方法参数是否符合注解规则
     * @param target 目标对象
     * @param methodName 方法名
     * @param args 方法参数
     * @throws IllegalArgumentException 校验失败时抛出
     */
    public static void validate(Object target, String methodName, Object... args) throws Exception {
        // 1. 获取方法对象
        Class<?>[] paramTypes = Arrays.stream(args)
            .map(arg -> arg == null ? Object.class : arg.getClass())
            .toArray(Class[]::new);
        Method method = target.getClass().getMethod(methodName, paramTypes);
        // 2. 获取方法参数列表(包含注解信息)
        Parameter[] parameters = method.getParameters();
        // 3. 遍历参数,执行校验
        for (int i = 0; i < parameters.length; i++) {
            Parameter param = parameters[i];
            Object arg = args[i]; // 当前参数值
            // 3.1 校验@NotNull 注解
            if (param.isAnnotationPresent(NotNull.class)) {
                NotNull notNull = param.getAnnotation(NotNull.class);
                if (arg == null) {
                    throw new IllegalArgumentException(notNull.message());
                }
            }
            // 3.2 校验@Range 注解(仅对数值类型生效)
            if (param.isAnnotationPresent(Range.class) && arg instanceof Number) {
                Range range = param.getAnnotation(Range.class);
                long value = ((Number) arg).longValue(); // 转为 long 统一处理
                if (value < range.min() || value > range.max()) {
                    throw new IllegalArgumentException(range.message());
                }
            }
        }
        // 4. 校验通过,执行方法
        method.invoke(target, args);
    }
}
步骤 4:测试参数校验效果
public class ValidatorTest {
    public static void main(String[] args) {
        UserService userService = new UserService();
        // 测试正常参数
        try {
            ParamValidator.validate(userService, "addUser", "user_001", 25);
        } catch (Exception e) {
            System.out.println("错误:" + e.getMessage());
        }
        // 测试 null 参数(userId 为 null)
        try {
            ParamValidator.validate(userService, "addUser", null, 25);
        } catch (Exception e) {
            System.out.println("错误:" + e.getMessage());
        }
        // 测试年龄超出范围(150 岁)
        try {
            ParamValidator.validate(userService, "addUser", "user_002", 150);
        } catch (Exception e) {
            System.out.println("错误:" + e.getMessage());
        }
        // 测试积分负数
        try {
            ParamValidator.validate(userService, "updatePoints", "user_003", -10);
        } catch (Exception e) {
            System.out.println("错误:" + e.getMessage());
        }
    }
}
执行结果
新增用户成功:userId=user_001, age=25 
错误:用户 ID 不能为空 
错误:年龄必须在 1-120 岁之间 
错误:积分不能为负数 
参数校验流程图解

文章配图

实战总结:自定义注解的设计原则与优化方向

1. 注解设计三要素

明确作用范围:用@Target严格限制注解的使用场景(如日志注解只用于方法);合理生命周期:需要反射解析的注解必须用@Retention(RUNTIME);属性精简实用:只保留必要的属性(如日志注解的description、校验注解的message),并设置合理默认值。

2. 反射解析优化技巧

缓存反射结果:Method、Parameter等对象的获取有性能开销,可通过ConcurrentHashMap缓存注解与方法的映射关系;批量处理注解:用getAnnotations()一次性获取所有注解,避免多次反射调用;异常友好提示:校验或日志失败时,异常信息要明确(如 "年龄必须在 1-120 岁之间" 而非 "参数错误")。

3. 与现有框架的对比

日志注解:Spring 的@Log、Lombok 的@Log4j2功能更完善,但自定义注解可灵活适配业务需求(如对接特定日志系统);参数校验:JSR-303 规范(javax.validation)提供了丰富的校验注解,但自定义注解可实现框架不支持的特殊校验(如手机号格式、身份证号规则)。

结语:注解 + 反射 = 代码的 "隐形翅膀"

通过本文的两个实战案例,你应该体会到:

注解是 "声明式编程" 的载体,让代码更简洁、意图更明确;反射是注解的 "执行引擎",让标记转化为实际逻辑。

这对组合在框架开发(如 Spring、MyBatis)中无处不在,掌握它们能让你从 "使用者" 升级为 "设计者"。

目录

  1. 前言:为什么需要自定义注解?
  2. 案例一:自定义日志注解@Log—— 自动记录方法调用轨迹
  3. 需求分析
  4. 步骤 1:定义@Log注解
  5. 步骤 2:创建使用@Log注解的业务类
  6. 步骤 3:实现注解解析器(核心逻辑)
  7. 步骤 4:测试日志注解效果
  8. 执行结果
  9. 日志注解执行流程图解
  10. 案例二:自定义参数校验注解 —— 告别重复的 if-else
  11. 需求分析
  12. 步骤 1:定义校验注解
  13. 1.1 @NotNull注解
  14. 1.2 @Range注解
  15. 步骤 2:创建使用校验注解的业务类
  16. 步骤 3:实现参数校验器(核心逻辑)
  17. 步骤 4:测试参数校验效果
  18. 执行结果
  19. 参数校验流程图解
  20. 实战总结:自定义注解的设计原则与优化方向
  21. 1. 注解设计三要素
  22. 2. 反射解析优化技巧
  23. 3. 与现有框架的对比
  24. 结语:注解 + 反射 = 代码的 "隐形翅膀"
  • 💰 8折买阿里云服务器限时8折了解详情
  • Magick API 一键接入全球大模型注册送1000万token查看
  • 🤖 一键搭建Deepseek满血版了解详情
  • 一键打造专属AI 智能体了解详情
极客日志微信公众号二维码

微信扫一扫,关注极客日志

微信公众号「极客日志V2」,在微信中扫描左侧二维码关注。展示文案:极客日志V2 zeeklog

更多推荐文章

查看全部
  • 如何利用 AI 自动生成自动化脚本
  • SLAM Toolbox 机器人建图与定位技术指南
  • 基于 STM32 的智能家居环境监测系统设计
  • Qwen3.5-4B 微调实战:LLaMA-Factory 打造医疗 AI 助手
  • 基于 IP-Adapter 与 AnimateDiff 的 AI 动漫短剧工业化流水线
  • Docker 项目部署实战:后端、前端与数据库配置
  • ToClaw:零门槛体验 OpenClaw AI 桌面自动化能力
  • Python 异步爬虫与 K8S 弹性伸缩:构建高并发数据采集引擎
  • Ubuntu 20.04 在 VMware 中的安装与配置指南
  • Git 撤销上一次提交实战:reset、revert 与 amend 用法详解
  • C++ 基础:树上 LCA(最近公共祖先)求解方法
  • Faster Whisper v1.7 本地语音转录与字幕生成教程(含 AMD 支持)
  • 智能电视芯片架构对比:ARM Cortex-A73 与 A55 性能差异
  • Java 面试核心知识点与高频问题解析
  • Stable Diffusion WebUI Forge AI 绘画风格转换指南
  • Java 及 Spring 环境下 Redis 操作详解
  • MySQL 数据库核心操作:创建、修改、备份与连接排查
  • CSP-S 提高组专题:倍增算法思想与应用(四)
  • Android 密码输入框禁用粘贴与复制功能实现
  • Java 时间类(上):JDK7 及以前 Date、SimpleDateFormat、Calendar 详解

相关免费在线工具

  • Keycode 信息

    查找任何按下的键的javascript键代码、代码、位置和修饰符。 在线工具,Keycode 信息在线工具,online

  • Escape 与 Native 编解码

    JavaScript 字符串转义/反转义;Java 风格 \uXXXX(Native2Ascii)编码与解码。 在线工具,Escape 与 Native 编解码在线工具,online

  • JavaScript / HTML 格式化

    使用 Prettier 在浏览器内格式化 JavaScript 或 HTML 片段。 在线工具,JavaScript / HTML 格式化在线工具,online

  • JavaScript 压缩与混淆

    Terser 压缩、变量名混淆,或 javascript-obfuscator 高强度混淆(体积会增大)。 在线工具,JavaScript 压缩与混淆在线工具,online

  • Base64 字符串编码/解码

    将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online

  • Base64 文件转换器

    将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online