本文基于前两篇文章扩展,看本文之前建议先看前两篇文章
Java动态代理实战(一):在Spring中实现代理接口无实现类注入
Java动态代理实战(二):dao接口无实现类且自动注册到Spring容器
看了前两篇文章发现在将代理dao注册到spring容器时需要知道具体的接口,不能用在实际项目中
真实项目肯定是希望像Mybaties一样,在应用入口处加上@MapperScan注解就自动扫描所有dao接口并注册到容器
我们首先定义注解@MyDaoScan
/**
* 扫描dao,并自动创建实现类
*
*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
// 使用import的方式导入
@Import(AutoDaoScanBeanDefinitionRegistrar.class)
public @interface MyDaoScan {
@AliasFor("value")
String[] basePackage() default {};
@AliasFor("basePackage")
String[] value() default {};
}
自动注册的核心代码 DaoScanClassPathBeanDefinitionScanner
public class DaoScanClassPathBeanDefinitionScanner extends ClassPathBeanDefinitionScanner {
public DaoScanClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry) {
super(registry);
}
@Override
protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent();
}
@Override
protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
Set<BeanDefinitionHolder> beanDefinitionHolders = super.doScan(basePackages);
processBeanDefinitions(beanDefinitionHolders);
//返回的是beandefinition
return beanDefinitionHolders;
}
private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitionHolders) {
beanDefinitionHolders.forEach(e ->{
GenericBeanDefinition definition = (GenericBeanDefinition) e.getBeanDefinition();
//这里是相当于调用BaseDaoProxyFactory中的setInterfaceClass()方法,详细的可以了解spring相关资料
definition.getPropertyValues().add("interfaceClass",definition.getBeanClassName());
// FactoryBean是一种特殊的Bean,这里的BeanClass是生成Bean实例的工厂,不是Bean本身。其返回的是该工厂Bean的getObject方法所返回的对象。
definition.setBeanClass(BaseDaoProxyFactory.class);
definition.setAutowireMode(GenericBeanDefinition.AUTOWIRE_BY_TYPE);
});
}
}
AutoDaoScanBeanDefinitionRegistrar
public class AutoDaoScanBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
// 获取MapperScan注解属性信息
AnnotationAttributes annotationAttributes = AnnotationAttributes.fromMap(importingClassMetadata.getAnnotationAttributes(HormDaoScan.class.getName()));
// 获取注解的属性值,拿到定义的扫描路径
String[] basePackages = annotationAttributes.getStringArray("basePackage");
// 使用自定义扫描器扫描
DaoScanClassPathBeanDefinitionScanner scanner = new DaoScanClassPathBeanDefinitionScanner(registry);
scanner.doScan(basePackages);
}
}
应用入口处加注解,实现扫描包动态代理
@SpringBootApplication
@ComponentScan(value = {"com.dddd.xxx.**"})//扫描包
@MyDaoScan(basePackage = "com.dddd.**.dao")//无实现类的dao扫描
public class App{
public static void main( String[] args ) {
SpringApplication.run(App.class, args);
}
}
使用方式
@Service("userTestService")
public class UserTestServiceImpl {
@Autowired
private UserTestDao testDao;
public void test() {
testDao.delete(1l);
}
}
代码抛砖引玉,作为正式产品差距比较大,直接使用会有很多问题,比如dao对象不是单例模式,每次操作数据库都会产生新的对象,如果并发量很大的话会浪费很多内存,甚至导致严重后果,各位可以根据自己具体需求实现功能