spring - 扫描自定义注解

spring - 扫描自定义注解

功能

自定义扫描注解,将扫描的bean加入到spring IOC容器中

@AnnoScan

/**
 * 自定义扫描注解,将扫描的bean加入到spring IOC容器中
 * @Date: 2024/4/18 16:26
 */
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Import(MyScannerRegistrar.class)
@Documented
public @interface AnnoScan {

    /**
     * 扫描指定包
     * @return
     */
    String[] basePackage() default {};

    /**
     * 指定扫描注解
     * @return
     */
    Class<? extends Annotation>[] annotations();

}

MyScanner

@Slf4j
public class MyScanner extends ClassPathBeanDefinitionScanner {


    public MyScanner(BeanDefinitionRegistry registry, Class<? extends Annotation>... annotations) {
        super(registry);
        for (Class<? extends Annotation> annotation : annotations) {
            addIncludeFilter(new AnnotationTypeFilter(annotation));
        }
    }

    @Override
    protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
        Set<BeanDefinitionHolder> beanDefinitionHolders = super.doScan(basePackages);
        if (!CollectionUtils.isEmpty(beanDefinitionHolders)){
            for (BeanDefinitionHolder beanDefinitionHolder : beanDefinitionHolders) {
                GenericBeanDefinition beanDefinition = (GenericBeanDefinition) beanDefinitionHolder.getBeanDefinition();
                log.info("MyScanner scanner class name: {}", beanDefinition.getBeanClassName());
            }
        }
        return beanDefinitionHolders;
    }
}

MyScannerRegistrar

@Slf4j
public class MyScannerRegistrar implements ImportBeanDefinitionRegistrar {


    private static final String BASE_PACKAGE = "basePackage";
    private static final String ANNOTATIONS = "annotations";

    @Override
    public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry registry) {

        AnnotationAttributes annotationAttributes = AnnotationAttributes.fromMap(annotationMetadata.getAnnotationAttributes(AnnoScan.class.getName()));
        if (annotationAttributes == null) {
            throw new RuntimeException("未注定必须属性");
        }
        Class<? extends Annotation>[] annos = getScanAnnotations(annotationMetadata, annotationAttributes);
        String[] basePackage = getBasePackage(annotationMetadata, annotationAttributes);
        log.info("MyScanner scan packages: {}", basePackage);
        for (Class<? extends Annotation> anno : annos) {
            MyScanner myScanner = new MyScanner(registry, anno);
            int count = myScanner.scan(basePackage);
            log.info("MyScanner scan annotation 【@{}】 count: 【{}】", anno.getSimpleName(), count);
        }
    }

    private Class<? extends Annotation>[] getScanAnnotations(AnnotationMetadata annotationMetadata, AnnotationAttributes annotationAttributes) {
        Class<?>[] annos = annotationAttributes.getClassArray(ANNOTATIONS);
        if (annos == null || annos.length == 0) {
            throw new RuntimeException("未指定扫描注解类型");
        }
        return (Class<? extends Annotation>[]) annos;
    }

    private String[] getBasePackage(AnnotationMetadata annotationMetadata, AnnotationAttributes annotationAttributes) {
        String[] basePackages = annotationAttributes.getStringArray(BASE_PACKAGE);
        if (basePackages == null || basePackages.length == 0) {
            basePackages = new String[]{((StandardAnnotationMetadata) annotationMetadata).getIntrospectedClass().getPackage().getName()};
        }
        return basePackages;
    }
}

测试

扫描 @RemoteService注解类

@Target({ ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RemoteService {

    /**
     * 属性字段
     * @return
     */
    String value() default "";
}

@RemoteService
public class CarService {
}

结果:

MyScanner scan packages: com.example.bootnacos
MyScanner scanner class name: com.example.bootnacos.scan.CarService
MyScanner scan annotation 【@RemoteService】 count: 【1】
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用Spring AOP来通过自定义注解来实现切入点。首先,定义一个自定义注解,并在需要切入的方法上使用该注解,然后定义一个切面,并在切面中使用@Before 或 @Around 注解来拦截被标记的方法,最后,在Spring的配置文件中声明所需的切面,此时,所有被标记的方法都会被拦截。 ### 回答2: 使用Spring AOP切入自定义注解有以下几个步骤: 1. 在项目中引入Spring AOP的依赖,一般为spring-aop和spring-aspects。 2. 在配置文件中开启Spring AOP的自动代理功能。可以通过在XML配置文件中添加<aop:aspectj-autoproxy />或在Java配置文件中添加@EnableAspectJAutoProxy注解实现。 3. 创建一个切面类,用于定义切入的逻辑。这个类需要使用@Component或者其他Spring注解进行标识,以便Spring能够扫描到。 4. 在切面类的方法中,使用@Before、@After等注解定义切入点和具体的切入操作。例如,使用@Before注解定义在某个注解标记的方法执行之前切入的逻辑。 5. 在注解中定义自定义的切点。可以使用@Retention和@Target等元注解来配置注解的生命周期和使用范围。 6. 在目标类或方法上添加自定义的注解。例如,在一个Service类的某个方法上添加自定义注解。 7. 运行项目,Spring会根据配置自动代理目标类,当目标类或方法被调用时,切面类中定义的切入逻辑就会自动被执行。 8. 可以通过配置切入的顺序、通知的类型等来进一步细化切入的逻辑。 通过以上步骤,我们可以使用Spring AOP方便地切入自定义注解,实现对目标类或方法的增强、日志记录、权限控制等功能。 ### 回答3: 使用Spring AOP切入自定义注解需要以下几个步骤: 1. 定义一个自定义的注解。可以使用Java提供的`@interface`关键字创建一个注解,例如: ```java @Target(ElementType.METHOD) // 定义注解的作用范围为方法 @Retention(RetentionPolicy.RUNTIME) // 注解在运行时可见 public @interface CustomAnnotation { // 自定义注解的属性 } ``` 2. 创建一个切面类来处理注解。可以使用Spring提供的`@Aspect`注解来标记切面类,并在方法上使用`@Before`、`@After`等注解来定义切入点和增强逻辑。例如: ```java @Aspect @Component public class CustomAspect { @Before("@annotation(customAnnotation)") // 拦截带有CustomAnnotation注解的方法 public void beforeMethod(CustomAnnotation customAnnotation) { // 在方法执行前执行的逻辑 } } ``` 3. 配置Spring AOP。在Spring配置文件中添加AOP的配置,例如使用`<aop:aspectj-autoproxy>`标签开启自动代理,并指定切面类的包名,让Spring能够自动扫描并应用切面逻辑。 4. 在目标方法上使用自定义注解。在需要切入的方法上标记使用自定义注解,例如: ```java @CustomAnnotation public void doSomething() { // 方法的实际逻辑 } ``` 这样,在调用`doSomething()`方法时,Spring AOP会拦截到带有`@CustomAnnotation`注解的方法,并执行切面逻辑。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值