【Spring编程常见错误50例】03.依赖注入常见错误-上

1.多个实现类 如何匹配

在实际的开发中,我们会使用@Autowired 注解进行依赖注入对应的bean,但是如果我们依赖的是一个接口,有对应多个实现的话,就会出现异常。

java 复制代码
@RestController
public class DbController {

    @Autowired
    private DbService dbService;

    @RequestMapping(path = "/hiDb",method = RequestMethod.GET)
    public String hiScope() {
        dbService.db();
        return "";
    }

}
java 复制代码
@Service
public class MySQLDbService implements DbService{

    @Override
    public void db() {
        System.out.println("mysql init");
    }
}

@Service
public class HbaseDbService implements DbService{

    @Override
    public void db() {
        System.out.println("HbaseDB");
    }
}
java 复制代码
Field dbService in com.qxlx.spingboot.controller.DbController required a single bean, but 2 were found:
	- hbaseDbService: defined in file [/Users/qxlx/work/qxlx/qxlx/spring/target/classes/com/qxlx/spingboot/service/HbaseDbService.class]
	- mySQLDbService: defined in file [/Users/qxlx/work/qxlx/qxlx/spring/target/classes/com/qxlx/spingboot/service/MySQLDbService.class]

原理分析

核心源码解析

AutowiredAnnotationBeanPostProcessor

java 复制代码
	public PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName) {
	     // 找出依赖的字段和方法 构建元数据 通过递归的方式进行
		InjectionMetadata metadata = findAutowiringMetadata(beanName, bean.getClass(), pvs);
		// 完成对方法或者字段的注入 执行注入 将依赖注入到bean
		metadata.inject(bean, beanName, pvs);
		return pvs;
	}
java 复制代码
	if (matchingBeans.size() > 1) {
		// 根据1.优先级@Primay来决策 2.@prority 3.bean名字
		autowiredBeanName = determineAutowireCandidate(matchingBeans, descriptor);
		if (autowiredBeanName == null) {
			// @Autowired是必须注入的, 注解的属性类型并不是可以接受的多个Bean类型
			if (isRequired(descriptor) || !indicatesMultipleBeans(type)) {
				return descriptor.resolveNotUnique(descriptor.getResolvableType(), matchingBeans);
			}
			else {
				return null;
			}
		}
	}

其实就是匹配多个条件,都不满足的时候,就抛出异常。因为没有办法决定使用那一个bean

解决方案

第一种 要么就是直接指定。

java 复制代码
    @Autowired
    private DbService mySQLDbService;

第二种指定

java 复制代码
	 @Autowired
	 @Qualifier("mySQLDbService")
	 private DbService dbService;

2.显示依赖bean首字母大小写 demo

但是如果我们故意写错了,回出现异常。

java 复制代码
	 @Autowired
	 @Qualifier("MySQLDbService")
	 private DbService dbService;

异常信息

java 复制代码
No qualifying bean of type 'com.qxlx.spingboot.service.DbService' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true), @org.springframework.beans.factory.annotation.Qualifier(value=MySQLDbService)}

具体原因

java 复制代码
public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) {
	// 如果是注解BD
	if (definition instanceof AnnotatedBeanDefinition) {
		// 查看bean是否有有name 有返回 
		String beanName = determineBeanNameFromAnnotation((AnnotatedBeanDefinition) definition);
		if (StringUtils.hasText(beanName)) {
			// Explicit bean name found.
			return beanName;
		}
	}
	// 设置一个默认的名称
	// Fallback: generate a unique default bean name.
	return buildDefaultBeanName(definition, registry);
}
java 复制代码
protected String buildDefaultBeanName(BeanDefinition definition) {
	// 获取BeanClassName
	String beanClassName = definition.getBeanClassName();
	Assert.state(beanClassName != null, "No bean class name set");
	// 获取类名
	String shortClassName = ClassUtils.getShortName(beanClassName);
	// 生成类名
	return Introspector.decapitalize(shortClassName);
}
java 复制代码
public static String decapitalize(String name) {
	// 基本验证
    if (name == null || name.length() == 0) {
        return name;
    }
    // 如果类名第一个和第二个都是大写 返回原来的类名
    // 比如 如果是MYSQLService 返回的就是MYSQLService
    if (name.length() > 1 && Character.isUpperCase(name.charAt(1)) &&
                    Character.isUpperCase(name.charAt(0))){
        return name;
    }
    char[] chars = name.toCharArray();
    //将第一个字母小写
    // 比如MySQLService 返回的就是mySQLService
    chars[0] = Character.toLowerCase(chars[0]);
    return new String(chars);
}

根据上述源码的分析,可以找到,在生成一个类名的时候,其实是有几个原则

1.如果类名提供了,那么按照对应的名称查找。

2.如果类名没有提供,并且前两个字母都是大写,那么默认就是原来的类名

3.类名是第一个字母小写。

掌握了上述原理,就可以轻松应对这些问题了。

3.引用内部类的bean类名

通过上面的案例,可以知道了beanName的规则,但是对于内部类来说。

java 复制代码
@Service
public class MySQLDbService implements DbService{

    @Override
    public void db() {
        System.out.println("mysql init");
    }

    @Service
    public static class MySQLDbServiceImpl implements DbService{
        @Override
        public void db() {
            System.out.println("mysql init");
        }
    }
}
java 复制代码
    @Autowired
    @Qualifier("MySQLDbService.MySQLDbServiceImpl")
    private DbService dbService;

发现出现错误。

java 复制代码
No qualifying bean of type 'com.qxlx.spingboot.service.DbService' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true), @org.springframework.beans.factory.annotation.Qualifier(value=MySQLDbService.MySQLDbServiceImpl)}

原理分析

在生成类名的时候,有一个方法

java 复制代码
String shortClassName = ClassUtils.getShortName(beanClassName);

public static String getShortName(String className) {
	Assert.hasLength(className, "Class name must not be empty");
	int lastDotIndex = className.lastIndexOf(PACKAGE_SEPARATOR);
	int nameEndIndex = className.indexOf(CGLIB_CLASS_SEPARATOR);
	if (nameEndIndex == -1) {
		nameEndIndex = className.length();
	}
	String shortName = className.substring(lastDotIndex + 1, nameEndIndex);
	shortName = shortName.replace(NESTED_CLASS_SEPARATOR, PACKAGE_SEPARATOR);
	return shortName;
}
相关推荐
IT_Octopus15 分钟前
多线程下使用缓存+锁Lock, 出现“锁失效” + “缓存未命中竞争”的缓存击穿情况,双重检查缓存解决问题
java·spring·缓存
简诚25 分钟前
PPHGNetV2源代码解析
python·深度学习·机器学习
阿幸软件杂货间35 分钟前
PPT转图片拼贴工具 v3.0
python·opencv·计算机视觉·powerpoint
雨白39 分钟前
Fragment 最佳实践:兼容手机和平板的简易新闻应用
android
洞见前行42 分钟前
Java反射机制详细指南
android
struggle20251 小时前
LLMControlsArm开源程序是DeepSeek 控制熊猫机械臂
人工智能·python·cmake·jupyternotebook·deepseek
Teln_小凯1 小时前
Python读取阿里法拍网的html+解决登录cookie
开发语言·python·html
点云SLAM1 小时前
PyTorch中matmul函数使用详解和示例代码
人工智能·pytorch·python·深度学习·计算机视觉·矩阵乘法·3d深度学习
花开月满西楼2 小时前
保姆级【快数学会Android端“动画“】+ 实现补间动画和逐帧动画!!!
android·前端·android studio
这儿有一堆花2 小时前
打造你的 Android 图像编辑器:深入解析 PhotoEditor 开源库
android·开源