JAVA重点基础、进阶知识及易错点总结(35)注解与反射

张开发
2026/4/5 22:25:58 15 分钟阅读

分享文章

JAVA重点基础、进阶知识及易错点总结(35)注解与反射
Java 巩固进阶 · 第 35 天主题注解与反射结合 —— 让注解活起来 进度概览继昨天学习注解定义之后今天进入注解的核心应用场景注解 反射。单独的注解只是标签只有结合反射读取并处理才能产生实际逻辑。这是 Spring、MyBatis、JUnit 等框架的底层核心。 核心价值框架原理理解 SpringAutowired注入、MyBatisSelect映射、SpringMVCRequestMapping绑定的底层实现。动态逻辑运行时根据注解信息动态执行代码实现配置即代码。工具开发掌握自定义校验框架、简易 IOC 容器、自动化日志切面的实现思路。面试高频反射读取注解流程、注解处理器APT概念是高级开发常考题。一、核心原理反射读取注解信息 1. 反射获取注解的 APIClass?clazzMyClass.class;// ✅ 1. 判断是否有指定注解if(clazz.isAnnotationPresent(MyAnnotation.class)){// 有注解}// ✅ 2. 获取注解实例读取属性值MyAnnotationannclazz.getAnnotation(MyAnnotation.class);Stringvalueann.value();// 读取注解属性// ✅ 3. 获取所有注解Annotation[]allAnnsclazz.getAnnotations();// ✅ 4. 获取方法/字段上的注解Methodmethodclazz.getMethod(doSomething);LoglogAnnmethod.getAnnotation(Log.class);⚠️关键前提注解必须用Retention(RetentionPolicy.RUNTIME)修饰否则运行时反射读取不到2. 处理流程图解定义注解 (Log) ↓ 标注在代码上 (UserService.createUser) ↓ 反射扫描类 (Class.forName) ↓ 获取方法 (getDeclaredMethods) ↓ 检查注解 (isAnnotationPresent) ↓ 读取属性 (getAnnotation) ↓ 执行逻辑 (打印日志/校验/注入)二、实战场景 1简易参数校验框架 ✅1. 定义校验注解// ✅ 定义 NotNull 注解Target(ElementType.FIELD)Retention(RetentionPolicy.RUNTIME)publicinterfaceNotNull{Stringmessage()default字段不能为空;}// ✅ 定义 Range 注解Target(ElementType.FIELD)Retention(RetentionPolicy.RUNTIME)publicinterfaceRange{intmin()default0;intmax()default100;}2. 标注在实体类上publicclassUser{NotNull(message用户名不能为空)privateStringname;Range(min1,max150)privateintage;// getter/setter}3. 实现校验器核心逻辑publicclassValidator{publicstaticvoidvalidate(Objectobj)throwsException{Class?clazzobj.getClass();// 1. 获取所有字段for(Fieldfield:clazz.getDeclaredFields()){field.setAccessible(true);Objectvaluefield.get(obj);// 2. 检查 NotNullif(field.isAnnotationPresent(NotNull.class)){NotNullannfield.getAnnotation(NotNull.class);if(valuenull||.equals(value)){thrownewIllegalArgumentException(ann.message());}}// 3. 检查 Rangeif(field.isAnnotationPresent(Range.class)){Rangeannfield.getAnnotation(Range.class);if(valueinstanceofInteger){intval(Integer)value;if(valann.min()||valann.max()){thrownewIllegalArgumentException(field.getName() 超出范围ann.min()-ann.max());}}}}}}// 测试UserusernewUser();user.setName();// 故意留空Validator.validate(user);// 抛异常用户名不能为空框架关联这就是Hibernate Validator (JSR-303)和 SpringValid的底层原理三、实战场景 2简易 AOP 日志切面 1. 回顾 Log 注解第 34 天定义Target(ElementType.METHOD)Retention(RetentionPolicy.RUNTIME)publicinterfaceLog{Stringvalue()default;Stringlevel()defaultINFO;}2. 实现日志拦截器模拟 Spring AOPpublicclassLogInterceptor{// ✅ 核心方法处理带有 Log 注解的方法publicstaticvoidinvokeWithLog(Objecttarget,Methodmethod,Object[]args)throwsException{// 1. 检查方法是否有 Log 注解if(method.isAnnotationPresent(Log.class)){LoglogAnnmethod.getAnnotation(Log.class);// 2. 前置日志System.out.println(【日志】开始执行logAnn.value() 级别logAnn.level());longstartSystem.currentTimeMillis();// 3. 执行目标方法method.setAccessible(true);Objectresultmethod.invoke(target,args);// 4. 后置日志longcostSystem.currentTimeMillis()-start;System.out.println(【日志】执行完成耗时costms);return;}// 无注解直接执行method.setAccessible(true);method.invoke(target,args);}}// 测试UserServiceservicenewUserService();Methodmethodservice.getClass().getMethod(createUser,String.class);LogInterceptor.invokeWithLog(service,method,newObject[]{Alice});框架关联Spring AOP 的Aspect本质也是扫描注解 动态代理 反射调用。四、实战场景 3简易依赖注入IOC 雏形1. 定义 MyAutowired 注解Target(ElementType.FIELD)Retention(RetentionPolicy.RUNTIME)publicinterfaceMyAutowired{Stringvalue()default;// 指定 Bean 名称}2. 实现简易容器publicclassSimpleContainer{privatestaticMapString,ObjectbeansnewHashMap();// ✅ 注册 BeanpublicstaticvoidregisterBean(Objectbean){Stringnamebean.getClass().getSimpleName();beans.put(name,bean);}// ✅ 注入依赖publicstaticvoidinjectDependencies(Objectbean)throwsException{Class?clazzbean.getClass();for(Fieldfield:clazz.getDeclaredFields()){// 1. 检查是否有 MyAutowiredif(field.isAnnotationPresent(MyAutowired.class)){field.setAccessible(true);// 2. 获取依赖类型Class?fieldTypefield.getType();StringbeanNamefieldType.getSimpleName();// 3. 从容器获取 BeanObjectdependencybeans.get(beanName);if(dependencynull){thrownewRuntimeException(未找到依赖 Bean: beanName);}// 4. 注入字段field.set(bean,dependency);}}}}3. 测试注入// 1. 定义 DaoclassUserDao{publicvoidsave(){System.out.println(UserDao.save);}}// 2. 定义 ServiceclassUserService{MyAutowiredprivateUserDaouserDao;// 自动注入publicvoidcreateUser(){userDao.save();}}// 3. 启动容器UserDaodaonewUserDao();UserServiceservicenewUserService();SimpleContainer.registerBean(dao);SimpleContainer.registerBean(service);SimpleContainer.injectDependencies(service);// 执行注入// 4. 调用service.createUser();// 输出UserDao.saveuserDao 已被自动赋值框架关联这就是Spring IOC 容器的核心逻辑扫描 → 创建 Bean → 扫描字段 → 注入依赖五、 今日实战任务注解框架实战任务 1实现参数校验器/** * 要求 * 1. 定义 NotNull 和 Range 注解 * 2. 创建 User 类字段上添加注解 * 3. 实现 Validator.validate() 方法 * 4. 测试传入合法/非法数据观察是否抛异常 * * 挑战 * - 支持嵌套对象校验如 User 中有 Address 字段 * - 支持自定义校验规则接口 */任务 2实现简易日志切面/** * 要求 * 1. 使用第 34 天定义的 Log 注解 * 2. 实现 LogInterceptor.invokeWithLog() 方法 * 3. 统计方法执行耗时 * 4. 测试调用带注解和不带注解的方法对比输出 * * 思考 * - 这种方式需要手动调用拦截器如何用动态代理自动拦截 * - 提示结合第 33 天的动态代理知识 */任务 3实现简易 IOC 容器/** * 要求 * 1. 定义 MyAutowired 注解 * 2. 实现 SimpleContainer 的 registerBean 和 injectDependencies * 3. 创建 Service Dao 结构测试依赖注入 * 4. 验证注入后的字段是否不为 null * * 挑战 * - 处理循环依赖问题A 依赖 BB 依赖 A * - 支持按类型注入不只按名称 */任务 4探索 Spring 源码/** * 要求 * 1. 打开 Spring 项目或查看在线源码 * 2. 找到 Autowired 注解定义查看其元注解 * 3. 找到 ClassUtils 或 ReflectionUtils 类查看 Spring 如何读取注解 * 4. 记录Spring 用了哪些反射工具方法 * * 提示 * 关注 AnnotatedElementUtils 类 */ 第 35 天 · 核心总结极简背诵版反射读取注解clazz.isAnnotationPresent(Ann.class)clazz.getAnnotation(Ann.class)method.getAnnotation(Ann.class)// ⚠️ 必须 Retention(RUNTIME)三大应用场景参数校验NotNull/Range → 反射检查字段值 日志切面Log → 反射调用方法前后增强 依赖注入Autowired → 反射设置字段值框架原理映射Hibernate Validator ← 参数校验 Spring AOP ← 日志切面 Spring IOC ← 依赖注入性能注意✅ 缓存反射对象Class/Method/Field✅ 缓存注解信息避免重复读取❌ 避免在高频循环中反射读取注解明天预告️Lombok 实战 阶段总结—— 提升开发效率的利器Lombok 核心注解Data/Builder/Slf4jLombok 原理注解处理器 APT注意事项与避坑指南第 3 阶段总结设计模式与注解知识脑图准备好了吗明天我们用工具解放双手并结束本阶段学习 ✨

更多文章