JAVA 注解(Annotation):从原理到实战应用

JAVA 注解(Annotation):从原理到实战应用

JAVA 注解(Annotation):从原理到实战应用

在这里插入图片描述

1.1 本章学习目标与重点

💡 掌握注解的核心概念与分类,理解注解在Java开发中的核心价值。
💡 熟练使用JDK内置注解,掌握自定义注解的定义、解析与使用流程。
💡 掌握注解的元注解配置方式,理解不同元注解对自定义注解的约束作用。
💡 结合反射机制实现注解的实战应用,掌握注解在框架开发中的核心用法。
⚠️ 本章重点是 自定义注解的开发流程注解与反射结合的实战应用,这是Java高级开发与框架设计的必备技能。

1.2 注解的核心概念与价值

1.2.1 什么是注解

💡 注解(Annotation) 是Java 5引入的一种特殊标记,它可以在编译期、类加载期、运行时被读取,并执行相应的处理逻辑。注解本身不直接影响代码的执行逻辑,而是通过元数据的方式为程序提供额外信息,这些信息可以被编译器、虚拟机或自定义的注解处理器解析和使用。

注解的本质是一个继承了 java.lang.annotation.Annotation 接口的特殊接口,我们定义的每一个注解,最终都会被编译器生成对应的接口实现类,供程序在运行时调用。

1.2.2 注解的核心价值

在传统Java开发中,我们需要通过配置文件(如XML)来描述程序的元数据信息,而注解的出现,让元数据可以直接嵌入到代码中,实现了代码与配置的一体化。注解的核心价值主要体现在以下几个方面:

  1. 简化配置:替代XML等外部配置文件,减少配置文件的数量,降低维护成本。
  2. 增强代码可读性:元数据与代码紧密结合,开发者可以直接在代码中看到配置信息。
  3. 编译期检查:部分注解可以在编译期进行语法检查,提前发现潜在问题。
  4. 框架开发基石:是Spring、MyBatis等主流框架的核心技术,用于实现依赖注入、AOP、ORM映射等功能。

1.2.3 注解的分类

根据注解的使用范围和生命周期,我们可以将注解分为以下三类:

分类标准具体类型特点典型示例
JDK内置注解基本注解用于修饰代码元素(类、方法、属性等)@Override@Deprecated@SuppressWarnings
元注解注解的注解用于修饰自定义注解,约束注解的使用范围和生命周期@Target@Retention@Documented@Inherited
自定义注解业务注解开发者根据业务需求自行定义的注解,用于实现特定功能@RequestMapping@Autowired@Test

✅ 核心结论:注解是一种元数据标记,本身不执行任何逻辑,需要通过注解处理器(如反射、APT工具)解析后,才能发挥实际作用。

1.3 JDK内置注解详解

JDK提供了三个最基础的内置注解,用于辅助代码开发,这三个注解也是我们日常开发中使用频率最高的注解。

1.3.1 @Override:重写检查注解

💡 作用:用于标记方法是重写自父类或接口的方法,编译器会检查该方法的签名是否与父类完全一致,如果不一致则编译报错。
⚠️ 注意事项:该注解只能修饰方法,且只能用于重写场景,不能用于重载。

代码实操
/** * @Override 注解使用示例 */classParentClass{publicvoidsayHello(){System.out.println("父类的sayHello方法");}}classChildClassextendsParentClass{// 使用@Override标记该方法是重写自父类@OverridepublicvoidsayHello(){System.out.println("子类重写的sayHello方法");}// 错误示例:方法签名与父类不一致,编译报错// @Override// public void sayHello(String name) {// System.out.println("Hello " + name);// }}publicclassOverrideDemo{publicstaticvoidmain(String[] args){ParentClass child =newChildClass(); child.sayHello();// 输出:子类重写的sayHello方法}}

1.3.2 @Deprecated:过时标记注解

💡 作用:用于标记类、方法、属性等元素已过时,不推荐使用。编译器会对使用该元素的代码发出警告,提示开发者更换新的API。
⚠️ 注意事项:该注解标记的元素仍然可以使用,但存在被移除的风险,建议开发者使用替代方案。

代码实操
/** * @Deprecated 注解使用示例 */publicclassDeprecatedDemo{// 标记该方法已过时@DeprecatedpublicstaticvoidoldMethod(){System.out.println("这个方法已经过时了,请使用newMethod替代");}// 新的替代方法publicstaticvoidnewMethod(){System.out.println("这是新的推荐方法");}publicstaticvoidmain(String[] args){// 使用过时方法,编译器会发出警告oldMethod();// 使用推荐方法newMethod();}}

1.3.3 @SuppressWarnings:抑制警告注解

💡 作用:用于抑制编译器产生的特定类型警告,让代码编译时不再输出警告信息。
⚠️ 注意事项:该注解需要传入参数,指定要抑制的警告类型,常用参数有 unchecked(抑制未检查转换警告)、deprecation(抑制过时元素警告)、all(抑制所有警告)。

代码实操
importjava.util.ArrayList;importjava.util.List;/** * @SuppressWarnings 注解使用示例 */publicclassSuppressWarningsDemo{publicstaticvoidmain(String[] args){// 抑制未检查转换警告@SuppressWarnings("unchecked")List<String> list =newArrayList();// 原始类型ArrayList,默认会产生警告 list.add("Hello Annotation");System.out.println(list.get(0));// 抑制过时方法警告@SuppressWarnings("deprecation")voidtest(){DeprecatedDemo.oldMethod();// 调用过时方法,不会产生警告}test();}}

✅ 核心结论:JDK内置注解主要用于代码检查和提示,帮助开发者写出更规范的代码,提升代码质量。

1.4 元注解详解

元注解是用于修饰自定义注解的注解,JDK提供了四个标准元注解,它们分别从不同维度约束自定义注解的行为。

1.4.1 @Target:注解目标约束

💡 作用:指定自定义注解可以修饰的代码元素类型,如类、方法、属性等。
⚠️ 注意事项:如果不指定 @Target,则该注解可以修饰所有类型的代码元素。

@Target 的参数是一个 ElementType 枚举数组,常用枚举值如下:

枚举值可修饰的代码元素
TYPE类、接口、枚举
METHOD方法
FIELD属性(成员变量)
PARAMETER方法参数
CONSTRUCTOR构造方法
LOCAL_VARIABLE局部变量
ANNOTATION_TYPE注解

1.4.2 @Retention:注解生命周期约束

💡 作用:指定注解的生命周期,即注解在哪个阶段有效。
⚠️ 注意事项:这是自定义注解必须指定的元注解,否则注解默认在编译期就会被丢弃,无法在运行时通过反射获取。

@Retention 的参数是一个 RetentionPolicy 枚举值,三种枚举值对应注解的三个生命周期:

枚举值生命周期特点适用场景
SOURCE源码阶段注解仅存在于源码中,编译成字节码后被丢弃编译器检查,如 @Override
CLASS字节码阶段注解存在于源码和字节码中,类加载时被丢弃类加载器处理,如字节码增强
RUNTIME运行时阶段注解存在于源码、字节码和运行时,可通过反射获取框架开发,如Spring注解

1.4.3 @Documented:文档生成标记

💡 作用:指定该注解会被 javadoc 工具提取到API文档中。
⚠️ 注意事项:默认情况下,注解不会出现在API文档中,添加该元注解后,注解信息会被包含在文档里。

1.4.4 @Inherited:注解继承标记

💡 作用:指定该注解可以被子类继承,如果父类使用了该注解,子类会自动继承该注解。
⚠️ 注意事项:该注解仅对类级别注解有效,对方法、属性等注解无效。

代码实操:元注解使用示例
importjava.lang.annotation.*;/** * 自定义注解 + 元注解使用示例 * 该注解用于标记类的作者信息 */// 该注解只能修饰类@Target(ElementType.TYPE)// 注解在运行时有效,可通过反射获取@Retention(RetentionPolicy.RUNTIME)// 注解信息会被javadoc提取@Documented// 注解可以被子类继承@Inheritedpublic@interfaceAuthor{// 注解属性:作者姓名,必填Stringname();// 注解属性:作者邮箱,可选,默认值为空字符串Stringemail()default"";}

✅ 核心结论:元注解是自定义注解的规则约束器,通过元注解可以精准控制自定义注解的使用范围和生命周期。

1.5 自定义注解的开发流程

自定义注解是注解开发的核心,掌握自定义注解的定义、使用和解析流程,是实现注解驱动开发的关键。

1.5.1 步骤1:定义自定义注解

定义自定义注解的语法格式与定义接口类似,只是在 interface 关键字前加了一个 @ 符号。注解中可以定义属性,属性的定义方式与接口方法类似,支持指定默认值。

语法格式
[元注解]public@interface 注解名称 {// 注解属性定义 属性类型 属性名()[default 默认值];}
代码实操:定义两个常用自定义注解
importjava.lang.annotation.*;/** * 注解1:用于标记业务日志,记录方法的操作信息 */@Target(ElementType.METHOD)// 仅修饰方法@Retention(RetentionPolicy.RUNTIME)// 运行时有效public@interfaceLog{// 操作描述Stringdesc();// 操作类型,默认值为"OTHER"Stringtype()default"OTHER";}/** * 注解2:用于标记数据校验,校验方法参数的合法性 */@Target(ElementType.PARAMETER)// 仅修饰方法参数@Retention(RetentionPolicy.RUNTIME)// 运行时有效public@interfaceValidate{// 参数不能为空booleannotNull()defaulttrue;// 参数长度最小值,默认值为0intminLength()default0;}

1.5.2 步骤2:使用自定义注解

定义好自定义注解后,就可以在对应的代码元素上使用该注解,使用时需要为注解的必填属性赋值,可选属性可以使用默认值。

代码实操:使用自定义注解
/** * 自定义注解使用示例:用户服务类 */publicclassUserService{// 使用@Log注解标记方法,记录操作日志@Log(desc ="根据ID查询用户", type ="QUERY")publicStringgetUserById(// 使用@Validate注解标记参数,校验参数合法性@Validate(notNull =true, minLength =1)String userId){if(userId ==null|| userId.length()<1){return"参数不合法";}return"用户ID:"+ userId +",用户名:张三";}@Log(desc ="添加用户", type ="ADD")publicStringaddUser(@Validate(notNull =true, minLength =2)String username){if(username ==null|| username.length()<2){return"用户名长度不能小于2";}return"添加用户成功:"+ username;}}

1.5.3 步骤3:解析自定义注解

注解本身不会执行任何逻辑,必须通过注解解析器来读取注解信息,并执行相应的处理逻辑。在Java中,最常用的注解解析方式是反射机制

解析流程

① 获取目标类的 Class 对象。
② 获取目标方法或参数的注解对象。
③ 读取注解的属性值,执行自定义业务逻辑。

代码实操:反射解析自定义注解
importjava.lang.reflect.Method;importjava.lang.reflect.Parameter;/** * 注解解析器:解析@Log和@Validate注解 */publicclassAnnotationParser{/** * 解析@Log注解,输出方法操作日志 */publicstaticvoidparseLogAnnotation(Class<?> clazz){// 获取类中所有方法Method[] methods = clazz.getDeclaredMethods();for(Method method : methods){// 判断方法是否有@Log注解if(method.isAnnotationPresent(Log.class)){Log log = method.getAnnotation(Log.class);System.out.println("===== 方法操作日志 =====");System.out.println("方法名:"+ method.getName());System.out.println("操作描述:"+ log.desc());System.out.println("操作类型:"+ log.type());System.out.println("-----------------------");}}}/** * 解析@Validate注解,校验方法参数合法性 */publicstaticObjectinvokeAndValidate(Object obj,String methodName,Object... args)throwsException{// 获取目标方法Method method = obj.getClass().getDeclaredMethod(methodName,getParameterTypes(args));Parameter[] parameters = method.getParameters();// 校验参数for(int i =0; i < parameters.length; i++){Parameter param = parameters[i];if(param.isAnnotationPresent(Validate.class)){Validate validate = param.getAnnotation(Validate.class);Object arg = args[i];String paramName = param.getName();// 非空校验if(validate.notNull()&& arg ==null){thrownewIllegalArgumentException(paramName +" 不能为null");}// 长度校验if(arg instanceofString&&((String) arg).length()< validate.minLength()){thrownewIllegalArgumentException(paramName +" 长度不能小于"+ validate.minLength());}}}// 调用目标方法return method.invoke(obj, args);}// 获取参数类型数组privatestaticClass<?>[]getParameterTypes(Object... args){if(args ==null|| args.length ==0){returnnewClass[0];}Class<?>[] types =newClass[args.length];for(int i =0; i < args.length; i++){ types[i]= args[i].getClass();}return types;}// 测试方法publicstaticvoidmain(String[] args)throwsException{UserService userService =newUserService();// 解析@Log注解parseLogAnnotation(UserService.class);// 解析@Validate注解并调用方法System.out.println("正常调用:"+invokeAndValidate(userService,"getUserById","1001"));System.out.println("参数非法调用:"+invokeAndValidate(userService,"getUserById",""));}}

输出结果

===== 方法操作日志 ===== 方法名:getUserById 操作描述:根据ID查询用户 操作类型:QUERY ----------------------- ===== 方法操作日志 ===== 方法名:addUser 操作描述:添加用户 操作类型:ADD ----------------------- 正常调用:用户ID:1001,用户名:张三 Exception in thread "main" java.lang.IllegalArgumentException: arg0 长度不能小于1 

✅ 核心结论:自定义注解的开发流程是 定义 → 使用 → 解析,其中解析是核心步骤,反射是最常用的解析方式。

1.6 注解的高级应用场景

注解在Java开发中应用广泛,尤其是在框架开发和业务系统设计中,以下是几个典型的高级应用场景。

1.6.1 场景1:基于注解的AOP实现

💡 核心思想:通过注解标记需要增强的方法,结合动态代理实现AOP(面向切面编程),在方法执行前后添加通用逻辑(如日志、事务、权限校验)。

代码实操:注解+动态代理实现日志切面
importjava.lang.reflect.InvocationHandler;importjava.lang.reflect.Method;importjava.lang.reflect.Proxy;/** * 基于注解的AOP日志切面 */// 定义切面注解@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@interfaceLogAspect{}// 业务接口interfaceOrderService{@LogAspectvoidcreateOrder(String orderId);}// 业务实现类classOrderServiceImplimplementsOrderService{@OverridepublicvoidcreateOrder(String orderId){System.out.println("创建订单:"+ orderId);}}// 日志切面处理器classLogAspectHandlerimplementsInvocationHandler{privateObject target;publicLogAspectHandler(Object target){this.target = target;}@OverridepublicObjectinvoke(Object proxy,Method method,Object[] args)throwsThrowable{// 方法执行前:判断是否有@LogAspect注解if(method.isAnnotationPresent(LogAspect.class)){System.out.println("[前置日志] 方法 "+ method.getName()+" 开始执行,参数:"+ args[0]);}// 执行目标方法Object result = method.invoke(target, args);// 方法执行后if(method.isAnnotationPresent(LogAspect.class)){System.out.println("[后置日志] 方法 "+ method.getName()+" 执行完毕");}return result;}// 创建代理对象publicstatic<T>TcreateProxy(T target){return(T)Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(),newLogAspectHandler(target));}}// 测试类publicclassAnnotationAopDemo{publicstaticvoidmain(String[] args){OrderService orderService =newOrderServiceImpl();OrderService proxy =LogAspectHandler.createProxy(orderService); proxy.createOrder("ORDER_20260129");}}

输出结果

[前置日志] 方法 createOrder 开始执行,参数:ORDER_20260129 创建订单:ORDER_20260129 [后置日志] 方法 createOrder 执行完毕 

1.6.2 场景2:基于注解的ORM映射

💡 核心思想:通过注解标记实体类与数据库表的映射关系,结合反射实现自动生成SQL语句,简化数据库操作。

代码实操:注解实现ORM映射
importjava.lang.annotation.*;/** * 基于注解的ORM映射示例 */// 表映射注解@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@interfaceTable{Stringname();}// 字段映射注解@Target(ElementType.FIELD)@Retention(RetentionPolicy.RUNTIME)@interfaceColumn{Stringname();Stringtype()default"VARCHAR";intlength()default255;}// 实体类@Table(name ="t_user")classUser{@Column(name ="user_id", type ="INT", length =10)privateInteger userId;@Column(name ="user_name", type ="VARCHAR", length =50)privateString userName;@Column(name ="age", type ="INT", length =3)privateInteger age;// 省略getter/setter方法publicIntegergetUserId(){return userId;}publicvoidsetUserId(Integer userId){this.userId = userId;}publicStringgetUserName(){return userName;}publicvoidsetUserName(String userName){this.userName = userName;}publicIntegergetAge(){return age;}publicvoidsetAge(Integer age){this.age = age;}}// ORM工具类:生成SQL语句classOrmUtil{publicstaticStringgenerateSelectSql(Class<?> clazz){// 获取表名Table table = clazz.getAnnotation(Table.class);if(table ==null){thrownewIllegalArgumentException("实体类未添加@Table注解");}String tableName = table.name();// 获取字段名StringBuilder columns =newStringBuilder();java.lang.reflect.Field[] fields = clazz.getDeclaredFields();for(java.lang.reflect.Field field : fields){Column column = field.getAnnotation(Column.class);if(column !=null){ columns.append(column.name()).append(",");}}// 移除最后一个逗号 columns.deleteCharAt(columns.length()-1);// 生成查询SQLreturn"SELECT "+ columns +" FROM "+ tableName;}publicstaticvoidmain(String[] args){System.out.println("生成的查询SQL:"+generateSelectSql(User.class));}}

输出结果

生成的查询SQL:SELECT user_id,user_name,age FROM t_user 

1.6.3 场景3:基于注解的单元测试

💡 核心思想:通过注解标记测试方法,测试框架自动识别并执行这些方法,实现自动化测试。这正是JUnit框架的核心实现原理。

代码实操:简化版JUnit框架
importjava.lang.annotation.*;importjava.lang.reflect.Method;/** * 简化版JUnit框架:基于注解的单元测试 */// 测试方法注解@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@interfaceTestCase{}// 测试类classMathTest{@TestCasepublicvoidtestAdd(){int result =1+2;assert result ==3:"加法测试失败";System.out.println("testAdd 测试通过");}@TestCasepublicvoidtestMultiply(){int result =2*3;assert result ==6:"乘法测试失败";System.out.println("testMultiply 测试通过");}// 非测试方法,不会被执行publicvoidnormalMethod(){System.out.println("这是普通方法,不会被测试");}}// 测试运行器classTestRunner{publicstaticvoidrun(Class<?> testClass)throwsException{Object obj = testClass.getDeclaredConstructor().newInstance();Method[] methods = testClass.getDeclaredMethods();int passCount =0;int failCount =0;for(Method method : methods){if(method.isAnnotationPresent(TestCase.class)){try{ method.invoke(obj); passCount++;}catch(Exception e){ failCount++;System.out.println(method.getName()+" 测试失败:"+ e.getCause().getMessage());}}}System.out.println("\n===== 测试结果 ======");System.out.println("总测试数:"+(passCount + failCount));System.out.println("通过数:"+ passCount);System.out.println("失败数:"+ failCount);}publicstaticvoidmain(String[] args)throwsException{TestRunner.run(MathTest.class);}}

输出结果

testAdd 测试通过 testMultiply 测试通过 ===== 测试结果 ====== 总测试数:2 通过数:2 失败数:0 

1.7 注解的优缺点与最佳实践

1.7.1 注解的优点

  1. 简化开发:减少冗余配置,提高开发效率。
  2. 增强代码可读性:元数据与代码紧密结合,一目了然。
  3. 提高代码灵活性:通过注解可以动态控制程序行为,无需修改核心代码。
  4. 框架友好:是现代Java框架的标配,便于集成主流框架。

1.7.2 注解的缺点

  1. 侵入性强:注解直接嵌入代码,修改注解需要修改源码。
  2. 可读性差:过多的注解会导致代码臃肿,影响阅读。
  3. 性能损耗:运行时注解需要通过反射解析,存在一定的性能开销。
  4. 调试困难:注解的逻辑在运行时执行,调试时难以跟踪。

1.7.3 注解的最佳实践

  1. 合理选择注解生命周期:仅在需要反射解析时使用 RUNTIME,否则使用 SOURCECLASS,减少性能损耗。
  2. 避免过度使用注解:不要用注解替代所有配置,对于经常变化的配置,建议使用外部配置文件。
  3. 做好注解文档:为自定义注解添加详细的JavaDoc注释,说明注解的作用和使用方式。
  4. 结合设计模式使用:如结合工厂模式、代理模式,实现注解的灵活解析和扩展。
  5. 优先使用成熟框架的注解:如Spring的 @Autowired@RequestMapping,避免重复造轮子。

1.8 实战案例:基于注解的权限校验框架

1.8.1 需求分析

💡 实现一个轻量级的权限校验框架,要求:

  1. 通过注解标记需要权限校验的方法。
  2. 支持指定方法所需的权限角色。
  3. 在方法执行前自动校验用户权限,无权限则抛出异常。
  4. 框架具备良好的扩展性,支持自定义权限校验逻辑。

1.8.2 代码实现

importjava.lang.annotation.*;importjava.lang.reflect.Method;importjava.util.Arrays;importjava.util.HashSet;importjava.util.Set;/** * 基于注解的权限校验框架 */// 权限校验注解@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)public@interfaceRequiresPermission{// 所需权限角色String[]value();}// 用户上下文类:存储当前登录用户信息classUserContext{privatestaticThreadLocal<String> currentUser =newThreadLocal<>();privatestaticSet<String> userRoles =newHashSet<>();// 设置当前用户publicstaticvoidsetCurrentUser(String username,String... roles){ currentUser.set(username); userRoles.clear(); userRoles.addAll(Arrays.asList(roles));}// 获取当前用户角色publicstaticSet<String>getUserRoles(){return userRoles;}// 清除当前用户publicstaticvoidclear(){ currentUser.remove(); userRoles.clear();}}// 权限校验切面类classPermissionAspect{/** * 执行方法并校验权限 */publicstaticObjectinvoke(Object obj,String methodName,Object... args)throwsException{Method method = obj.getClass().getDeclaredMethod(methodName,getParameterTypes(args));// 权限校验if(method.isAnnotationPresent(RequiresPermission.class)){RequiresPermission annotation = method.getAnnotation(RequiresPermission.class);String[] requiredRoles = annotation.value();Set<String> userRoles =UserContext.getUserRoles();// 检查是否有匹配的角色boolean hasPermission =false;for(String role : requiredRoles){if(userRoles.contains(role)){ hasPermission =true;break;}}if(!hasPermission){thrownewSecurityException("无权限访问该方法,所需角色:"+Arrays.toString(requiredRoles));}}// 执行目标方法return method.invoke(obj, args);}privatestaticClass<?>[]getParameterTypes(Object... args){if(args ==null|| args.length ==0){returnnewClass[0];}Class<?>[] types =newClass[args.length];for(int i =0; i < args.length; i++){ types[i]= args[i].getClass();}return types;}}// 业务服务类classAdminService{@RequiresPermission({"ADMIN","SUPER_ADMIN"})publicStringdeleteUser(String userId){return"删除用户成功:"+ userId;}@RequiresPermission({"USER","ADMIN","SUPER_ADMIN"})publicStringqueryUser(String userId){return"查询用户信息:"+ userId;}}// 测试类publicclassPermissionFrameworkDemo{publicstaticvoidmain(String[] args)throwsException{AdminService adminService =newAdminService();// 测试1:管理员用户,拥有ADMIN角色UserContext.setCurrentUser("admin","ADMIN");System.out.println(PermissionAspect.invoke(adminService,"deleteUser","1001"));System.out.println(PermissionAspect.invoke(adminService,"queryUser","1001"));// 测试2:普通用户,只有USER角色UserContext.setCurrentUser("user","USER");System.out.println(PermissionAspect.invoke(adminService,"queryUser","1001"));try{PermissionAspect.invoke(adminService,"deleteUser","1001");}catch(Exception e){System.out.println(e.getCause().getMessage());}UserContext.clear();}}

输出结果

删除用户成功:1001 查询用户信息:1001 查询用户信息:1001 无权限访问该方法,所需角色:[ADMIN, SUPER_ADMIN] 

1.8.3 案例总结

✅ 这个权限校验框架综合运用了注解定义、反射解析、线程本地存储等技术,核心亮点如下:

  1. 使用 @RequiresPermission 注解标记需要权限校验的方法,配置简单。
  2. 通过 UserContext 存储当前用户角色,支持多线程环境。
  3. 利用反射解析注解,在方法执行前自动校验权限,无侵入性。
  4. 框架扩展性强,可通过修改 PermissionAspect 类实现自定义权限校验逻辑。

1.9 本章总结

  1. 注解是Java的一种元数据标记,本身不执行逻辑,需要通过解析器(反射、APT)处理后才能发挥作用。
  2. JDK内置注解用于代码检查,元注解用于约束自定义注解,自定义注解是业务开发的核心。
  3. 自定义注解的开发流程是 定义 → 使用 → 解析,反射是最常用的运行时解析方式。
  4. 注解是框架开发的基石,广泛应用于AOP、ORM、权限校验、单元测试等场景。
  5. 注解的优点是简化配置、增强可读性,缺点是侵入性强、有性能损耗,使用时需平衡利弊。
  6. 注解的最佳实践是“合理选择生命周期、避免过度使用、做好文档注释”,结合设计模式提升扩展性。

Read more

【2026 最新】Java 与 IntelliJ IDEA 详细下载安装教程 带图演示(Windows 版)

【2026 最新】Java 与 IntelliJ IDEA 详细下载安装教程 带图演示(Windows 版)

前言 Java 是全球最广泛使用的编程语言之一,适用于企业级应用、Android 开发、大数据处理和后端服务。而 IntelliJ IDEA(简称 IDEA)作为 JetBrains 公司推出的旗舰级 Java IDE,凭借卓越的智能代码补全、深度框架集成、强大的调试器和流畅的用户体验,被广大开发者誉为“最强 Java 开发工具”。 本教程专为 Windows 系统用户 编写,将手把手指导你完成 Java Development Kit (JDK) 和 IntelliJ IDEA 的下载、安装与基础配置,助你快速搭建专业的 Java 开发环境。 一、Java(JDK)下载与安装 💡 注意:开发 Java 程序需要安装 JDK(

By Ne0inhk
【JAVA 进阶】SpringBoot自动配置机制:从原理到实践的深度解析

【JAVA 进阶】SpringBoot自动配置机制:从原理到实践的深度解析

文章目录 * 前言 * 第一章 初识SpringBoot自动配置 * 1.1 自动配置的定义 * 1.2 自动配置的核心价值 * 1.2.1 降低开发门槛 * 1.2.2 提高开发效率 * 1.2.3 保证配置一致性 * 1.3 自动配置与传统Spring配置的对比 * 1.3.1 传统Spring Web配置(Spring 4.x及之前) * 1.3.2 SpringBoot自动配置实现 * 第二章 深入原理:SpringBoot自动配置是如何实现的 * 2.1 核心注解:@SpringBootApplication的“三位一体” * 2.1.1 @SpringBootConfiguration:标识配置类

By Ne0inhk
JAVA | 聚焦 OutOfMemoryError 异常

JAVA | 聚焦 OutOfMemoryError 异常

个人主页 文章专栏 在正文开始前,我想多说几句,也就是吐苦水吧…最近这段时间一直想写点东西,停下来反思思考一下。 心中万言,真正执笔时又不知先写些什么。通常这个时候,我都会随便写写,文风极像散文,形散意不散吧! 先说一下近况,最近参加了Mathorlab数学建模,作为一个大一的学生,第一次参加那么高强度的竞赛。深深的意识到自己的不足,天之大,不过蚍蜉撼树。我不过是渺小的沧海一粟。竟欲与苍天比高,不自量力、痴人说梦。梦醒了,还是得加油干…高考后,天真的以为自己不用学习了,后来发现,自己一辈子都要学习。害,挺难过的,也挺无助的。 我,出现在这里。可能你们以为我是理科生,工科女…哦不!我江苏高考纯文出身,这像极了案底。高二下之前,我的确是根正苗红的物化生。可惜,尔辈不能究物理。我无比后悔当时的决定,可是人是不会满足的,那时候不管学啥,我都会后悔。用现在的眼光,去埋怨当时的自己。

By Ne0inhk
Java 大视界 --Java 大数据在智能教育学习资源整合与知识图谱构建中的深度应用(406)

Java 大视界 --Java 大数据在智能教育学习资源整合与知识图谱构建中的深度应用(406)

Java 大视界 --Java 大数据在智能教育学习资源整合与知识图谱构建中的深度应用(406) * 引言: * 正文: * 一、智能教育的两大核心痛点与 Java 大数据的适配性 * 1.1 资源整合:42% 重复率背后的 “三大堵点” * 1.2 知识图谱:83% 学生面临 “知识衔接断层” * 1.3 Java 大数据的 “适配性优势”:为什么选 Java 不选其他? * 二、Java 大数据技术栈选型:贴合教育场景的 “最优解” * 2.1 选型三大核心原则 * 2.2 核心技术栈与场景适配性 * 三、核心方案设计:资源整合 + 知识图谱双引擎 * 3.1 整体架构:从

By Ne0inhk