前言:由于好奇看了一下《Spring注解Component原理源码解析》的博文,就自己简单写了一个demo,用于扩展自己的类交由spring来管理,emmm。。。。。应用场景的话就比如一些比较老的项目(非spring项目)想改造为spring项目,就可以将项目中某个顶级接口设定为@Component一样的功能。
- 首先声明我们的标记接口,他就相当于@Component
public interface MyService {
}
- 自定义我们的扫描器,就相当于我们的@ComponentScan(basepackage=“xx.xx…”)
import org.springframework.beans.factory.config.BeanDefinitionHolder;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.context.annotation.ClassPathBeanDefinitionScanner;
import org.springframework.core.type.filter.AssignableTypeFilter;
import java.util.Set;
/**
* 自定义扫描器
*/
public class TestClassPathBeanDefinitionScanner extends ClassPathBeanDefinitionScanner {
public TestClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry) {
// 不使用默认的过滤器
super(registry, false);
}
@Override
protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
System.out.println("doscan.....");
return super.doScan(basePackages);
}
/**
* 添加自己定义的过滤类型,可以是注解,可以是接口,可以是类
*/
protected void registerTypeFilter() {
addIncludeFilter(new AssignableTypeFilter(MyService.class));
}
}
- 扫描器的声明
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor;
import org.springframework.stereotype.Component;
/**
* 通过这里去实现自定义的扫描器
*/
@Component
public class TestBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {
@Override
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
TestClassPathBeanDefinitionScanner scanner = new TestClassPathBeanDefinitionScanner(registry);
scanner.registerTypeFilter();
// 这里配置扫描包最好写成配置
int scan = scanner.scan("com.platform.mst.helper");
System.out.println(scan);
}
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
}
}
以上就是我们自定义扫描类的实现,下面是一个测试类
4. 写一个我们的测试类
// 这个类我们不加@Component,但是需要implements我们定义的标记接口
public class TestService implements MyService {
public TestService() {
System.out.println("TestService------");
}
@MyLogger // 测试是否支持spring的切面
public String testAop(){
return "testAop----------";
}
}
下面是扩展测试我们定义交由spring管理的类是否支持切面:
import java.lang.annotation.*;
// 定义切面注解
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@Documented
public @interface MyLogger {
}
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
// 切面的实现
@Aspect
public class MyLoggerAspect implements MyService {
@Pointcut("@annotation(com.platform.mst.helper.service.MyLogger)")
public void sayings() {
}
@Before("sayings()")
public void sayHello() {
System.out.println("注解类型前置通知");
}
}
最后是我们的启动类:
import com.platform.mst.helper.service.TestService;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.transaction.annotation.EnableTransactionManagement;
/**
* 辅助模块
* @author lix
* @Date 2020/4/17
*/
@SpringBootApplication
@EnableEurekaClient
@MapperScan("com.platform.mst.sdk.database.mapper")
@EnableTransactionManagement //开启事务注解
public class MstHelperApplication {
public static void main(String[] args) {
ConfigurableApplicationContext run = SpringApplication.run(MstHelperApplication.class, args);
TestService bean = run.getBean(TestService.class);
System.out.println(bean.testAop());
}
}
启动日志:
2020-06-28 14:25:00.121 INFO 10556 --- [ main] c.p.mst.helper.MstHelperApplication : No active profile set, falling back to default profiles: default
doscan.....
2
2020-06-28 14:25:01.240 INFO 10556 --- [ main] o.s.cloud.context.scope.GenericScope : BeanFactory id=86821ff6-6bde-3f75-bde6-07bbed1221a1
.....不重要的日志信息省略....
TestService------
2020-06-28 14:25:08.983 INFO 10556 --- [ main] o.s.c.n.eureka.InstanceInfoFactory : Setting initial instance status as: STARTING
.....不重要的日志信息省略....
2020-06-28 14:25:10.628 INFO 10556 --- [ main] c.p.mst.helper.MstHelperApplication : Started MstHelperApplication in 13.826 seconds (JVM running for 14.763)
注解类型前置通知
testAop----------
2020-06-28 14:25:10.661 INFO 10556 --- [nfoReplicator-0] com.netflix.discovery.DiscoveryClient : DiscoveryClient_MST-HELPER/130.24.37.199:9090 - registration status: 204
2020-06-28 14:25:40.601 INFO 10556 --- [freshExecutor-0] com.netflix.discovery.DiscoveryClient : Disable delta property : false