千家信息网

@MapperScan扫描包里混有@Service等问题怎么办

发表于:2025-01-18 作者:千家信息网编辑
千家信息网最后更新 2025年01月18日,小编给大家分享一下@MapperScan扫描包里混有@Service等问题怎么办,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!@MapperScan扫描包混有@Service问题描述
千家信息网最后更新 2025年01月18日@MapperScan扫描包里混有@Service等问题怎么办

小编给大家分享一下@MapperScan扫描包里混有@Service等问题怎么办,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!

@MapperScan扫描包混有@Service

问题描述

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

解决办法一

把service包移走,方法可行

解决办法二

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

解决办法三

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

创建注解@MyMapperScan

里面的属性全部抄袭@MapperScan

MapperScannerRegistrar换成自己的

@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 nameGenerator() default BeanNameGenerator.class;  Class annotationClass() default Annotation.class;  Class markerInterface() default Class.class;   String sqlSessionTemplateRef() default "";  String sqlSessionFactoryRef() default "";    Class factoryBean() default MapperFactoryBean.class;  String lazyInitialization() default "";}

MyMapperScannerRegistrar扫描注册器

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

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 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 generatorClass = annoAttrs.getClass("nameGenerator");        if (!BeanNameGenerator.class.equals(generatorClass)) {            scanner.setBeanNameGenerator(BeanUtils.instantiateClass(generatorClass));        }        Class 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 basePackages = new ArrayList();        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的,就是在判断上面改动了一点点

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的包:

import org.mybatis.spring.annotation.MapperScan;

结果就异常:

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

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

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

import tk.mybatis.spring.annotation.MapperScan;

看完了这篇文章,相信你对"@MapperScan扫描包里混有@Service等问题怎么办"有了一定的了解,如果想了解更多相关知识,欢迎关注行业资讯频道,感谢各位的阅读!

0