好得很程序员自学网

<tfoot draggable='sEl'></tfoot>

@MapperScan扫描包里混有@Service等问题如何解决

@MapperScan扫描包混有@Service

问题描述

@MapperScan注解配置的一般是dao或者mapper的扫描包,一般用于数据库操作,里面类的一般都是接口,如果在dao层有其他接口,比如说@Service等就会报错

解决办法一

把service包移走,方法可行

解决办法二

不使用@MapperScan,在每个dao或者mapper里面加上注解@Mapper,方法可行

解决办法三

使用自定义注解,在mybatis的注解比较完善的情况下,就不用自己搞多少

创建注解@MyMapperScan

里面的属性全部抄袭@MapperScan

MapperScannerRegistrar换成自己的

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

@Retention (RetentionPolicy.RUNTIME)

@Target (ElementType.TYPE)

@Documented

@Import (MyMapperScannerRegistrar. class )

@Repeatable (MapperScans. class )

public @interface MapperScan {

   String[] value() default {};

   String[] basePackages() default {};

 

   Class<?>[] basePackageClasses() default {};

 

   Class<? extends BeanNameGenerator> nameGenerator() default BeanNameGenerator. class ;

   Class<? extends Annotation> annotationClass() default Annotation. class ;

   Class<?> markerInterface() default Class. class ;

 

   String sqlSessionTemplateRef() default "" ;

   String sqlSessionFactoryRef() default "" ;

  

   Class<? extends MapperFactoryBean> factoryBean() default MapperFactoryBean. class ;

   String lazyInitialization() default "" ;

}

MyMapperScannerRegistrar扫描注册器

这个类和mybatis的一模一样,唯一的不同就是MyClassPathMapperScanner是自己的扫描

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

public class MyMapperScannerRegistrar extends MapperScannerRegistrar {

     private ResourceLoader resourceLoader;

     /**

      * {@inheritDoc}

      */

     @Override

     public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {

         AnnotationAttributes annoAttrs = AnnotationAttributes.fromMap(importingClassMetadata.getAnnotationAttributes(MapperScan. class .getName()));

         //这个是自己的

         ClassPathMapperScanner scanner = new MyClassPathMapperScanner(registry);

         // this check is needed in Spring 3.1

         if (resourceLoader != null ) {

             scanner.setResourceLoader(resourceLoader);

         }

         Class<? extends Annotation> annotationClass = annoAttrs.getClass( "annotationClass" );

         if (!Annotation. class .equals(annotationClass)) {

             scanner.setAnnotationClass(annotationClass);

         }

         Class<?> markerInterface = annoAttrs.getClass( "markerInterface" );

         if (!Class. class .equals(markerInterface)) {

             scanner.setMarkerInterface(markerInterface);

         }

         Class<? extends BeanNameGenerator> generatorClass = annoAttrs.getClass( "nameGenerator" );

         if (!BeanNameGenerator. class .equals(generatorClass)) {

             scanner.setBeanNameGenerator(BeanUtils.instantiateClass(generatorClass));

         }

         Class<? extends MapperFactoryBean> mapperFactoryBeanClass = annoAttrs.getClass( "factoryBean" );

         if (!MapperFactoryBean. class .equals(mapperFactoryBeanClass)) {

             scanner.setMapperFactoryBean(BeanUtils.instantiateClass(mapperFactoryBeanClass));

         }

         scanner.setSqlSessionTemplateBeanName(annoAttrs.getString( "sqlSessionTemplateRef" ));

         scanner.setSqlSessionFactoryBeanName(annoAttrs.getString( "sqlSessionFactoryRef" ));

         List<String> basePackages = new ArrayList<String>();

         for (String pkg : annoAttrs.getStringArray( "value" )) {

             if (StringUtils.hasText(pkg)) {

                 basePackages.add(pkg);

             }

         }

         for (String pkg : annoAttrs.getStringArray( "basePackages" )) {

             if (StringUtils.hasText(pkg)) {

                 basePackages.add(pkg);

             }

         }

         for (Class<?> clazz : annoAttrs.getClassArray( "basePackageClasses" )) {

             basePackages.add(ClassUtils.getPackageName(clazz));

         }

         scanner.registerFilters();

         scanner.doScan(StringUtils.toStringArray(basePackages));

     }

     /**

      * {@inheritDoc}

      */

     @Override

     public void setResourceLoader(ResourceLoader resourceLoader) {

         this .resourceLoader = resourceLoader;

     }

}

MyClassPathMapperScanner

自己的扫描类基本也是mybatis的,就是在判断上面改动了一点点

?

1

2

3

4

5

6

7

8

9

10

11

12

public class MyClassPathMapperScanner extends ClassPathMapperScanner {

     public MyClassPathMapperScanner(BeanDefinitionRegistry registry) {

         super (registry);

     }

     @Override

     protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {

         boolean flag = super .isCandidateComponent(beanDefinition);

         //包名带有Mapper的才会被mybatis代理

         boolean mapper = beanDefinition.getBeanClassName().contains( "Mapper" );

         return flag && mapper;

     }

}

现在只需要用自己的扫描注解即可,用法和mybatis的一模一样

解决办法四

这个是针对第三点的,作者使用第三点的时候mybatis版本为3.4.6

mybatis-spring版本为1.3.2,spring版本为5.x

当mybatis版本为3.5.2的mybatis-spring版本为2.0.2的时候

MapperScannerRegistrar类扫描的方式发生了一点点变化,

还需要改写MapperScannerConfigurer类,其他的不变

完毕!

解决办法五

@MapperScan注解使用markerInterface或者annotationClass参数限制扫描的接口

@MapperScan包扫描的坑

在使用通用mapper执行查询时

由于不太注意顺手就导了spring的包:

?

1

import org.mybatis.spring.annotation.MapperScan;

结果就异常:

tk.mybatis.mapper.provider.base.BaseSelectProvider:xxxx

找了半天才发现是包的问题

应该导mybatis的MapperScan而不是spring中的包,正确的包名:

?

1

import tk.mybatis.spring.annotation.MapperScan;

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。

原文链接:https://blog.csdn.net/weixin_43328357/article/details/103993232

查看更多关于@MapperScan扫描包里混有@Service等问题如何解决的详细内容...

  阅读:14次