报错信息(节选):
org.springframework.beans.factory.BeanCreationException:
Error creating bean with name 'accountController':
Injection of resource dependencies failed; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException:
Error creating bean with name 'accountService': Unsatisfied dependency expressed through field 'accountMapper';
Error resolving class. Cause: org.apache.ibatis.type.TypeException: Could not resolve type alias 'Account'. Cause: java.lang.ClassNotFoundException: Cannot find class: Account
问题概述:
此问题主要是在XML中引用类型的时候,直接使用了Alias,而因为Springboot和Mybatis之间存在一些"小摩擦",结合的不够完美导致(本人用的springboot2.1.4版本依然未解决)。具体原因分析网上较多,可惜大部分对解决方案都语焉不详。本文将解决方案完整贴出来,方便后来者,同时也给自己做个备忘。
解决方案:
- 弃用Alias,直接在xml中需要引入Entity类型的地方直接引入类的全路径+类名。
网上找到的大部分都是这个方案,个人感觉这不算是个优雅的解决方案。特别是有些复杂的业务处理xml,返回类型也较多的场景,看着满目都是resultType='xxx.xxx.xxx.xxx.xxx.Xxxx'绝对不是一件让人愉快的事情,分分钟逼死强迫症。 - 手动设置SqlSessionFactoryBean, 设置VFS为SpringBootVFS。
新建一个配置类MybatisConfiguration,手动注入SqlSessionFactoryBean。
@Configuration
public class MybatisConfiguration {
// 数据源
@Resource
private DataSource dataSource;
// 资源文件(application.yml文件中mybatis下级资源)读取器
@Resource
private MybatisProperties properties;
/**
* 设置SqlSessionFactory, 并设置手动设置VFS为SpringBootVFS
*/
@Bean(name = "sqlSessionFactory")
public SqlSessionFactoryBean sqlSessionFactory() {
SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
bean.setDataSource(this.dataSource);
bean.setVfs(SpringBootVFS.class); // 关键,将vfs设置为SpringBootVFS
bean.setTypeAliasesPackage(this.properties.getTypeAliasesPackage()); // 既然手动了,别忘了把其他的配置一起加入
bean.setMapperLocations(this.properties.resolveMapperLocations());
return bean;
}
}
application.yml部分,喜欢用application.properties的同学自行转换:
mybatis:
#扫描Entity, 加入此设置方可在Mybatis的XML文件中使用实体类的Alias
typeAliasesPackage: com.teamsoft.springboot2.*.entity
#扫描Mybatis的XML文件,
mapperLocations: com/teamsoft/springboot2/*/mapper/*.xml
啥时候这个BUG被修复了,或者有更好的解决方案,欢迎告知。
网友评论