10. primary可以解决什么问题?

文章讲述了在Spring框架中,当一个接口有多重实现时,如何通过`primary`属性指定主bean,解决从容器中获取bean时的`NoUniqueBeanDefinitionException`异常。作者通过示例展示了如何配置primary属性并处理可能出现的问题。

存在的问题以及解决方案

直接上案例,通过案例来看技术是如何使用的:


  1. package com.javacode2018.lesson001.demo8;
  2. public class NormalBean {
  3. public interface IService{} //@1
  4. public static class ServiceA implements IService{} //@2
  5. public static class ServiceB implements IService{} //@3
  6. }

上面代码很简单,@1:定义了一个接口IService,@2和@3创建了两个类都实现了IService接口。

下面我们通过spring来定义ServiceA和ServiceB两个bean,配置文件(normalBean.xml)如下:


  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
  6. <bean id="serviceA" class="com.javacode2018.lesson001.demo8.NormalBean$ServiceA"/>
  7. <bean id="serviceB" class="com.javacode2018.lesson001.demo8.NormalBean$ServiceB"/>
  8. </beans>

来个测试用例来从spring容器中获取上面定义的bean对象,如下:


  1. package com.javacode2018.lesson001.demo8;
  2. import com.javacode2018.lesson001.demo5.IocUtils;
  3. import org.junit.Test;
  4. import org.springframework.context.support.ClassPathXmlApplicationContext;
  5. /**
  6. * 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!
  7. * bean元素的primary属性可以解决什么问题?
  8. */
  9. public class PrimaryTest {
  10. @Test
  11. public void normalBean() {
  12. String beanXml = "classpath:/com/javacode2018/lesson001/demo8/normalBean.xml";
  13. ClassPathXmlApplicationContext context = IocUtils.context(beanXml);
  14. //下面我们通过spring容器的T getBean(Class<T> requiredType)方法获取容器中对应的bean
  15. NormalBean.IService service = context.getBean(NormalBean.IService.class); //@1
  16. System.out.println(service);
  17. }
  18. }

注意@1的代码,从spring容器中在容器中查找NormalBean.IService.class类型的bean对象,我们来运行一下看看效果,部分输出如下:


  1. org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.javacode2018.lesson001.demo8.NormalBean$IService' available: expected single matching bean but found 2: serviceA,serviceB
  2. at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveNamedBean(DefaultListableBeanFactory.java:1180)
  3. at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveBean(DefaultListableBeanFactory.java:416)
  4. at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:349)
  5. at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:342)
  6. at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1126)

发生异常了,错误中有一段提示比较重要,如下:


  1. org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.javacode2018.lesson001.demo8.NormalBean$IService' available: expected single matching bean but found 2: serviceA,serviceB

这个详细说出了错误原因:spring容器中定义了2个bean,分别是serviceA和serviceB,这两个bean对象都实现了IService接口,而用例中我们想从容器中获取IService接口对应的bean,此时容器中有2个候选者(serviceA和serviceB)满足我们的需求,此时spring容器不知道如何选择,到底是返回serviceA呢还是返回serviceB呢?spring容器也懵逼了,所以报错了。

再来看一个通过setter方法注入的案例:


  1. package com.javacode2018.lesson001.demo8;
  2. public class SetterBean {
  3. public interface IService{} //@1
  4. public static class ServiceA implements IService{} //@2
  5. public static class ServiceB implements IService{} //@3
  6. private IService service;
  7. public void setService(IService service) {
  8. this.service = service;
  9. }
  10. }

下面我们通过xml来定义SetterBean,并且使用setter方式将IService注入到SetterBean中,配置如下:


  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
  6. <bean id="serviceA" class="com.javacode2018.lesson001.demo8.SetterBean$ServiceA"/>
  7. <bean id="serviceB" class="com.javacode2018.lesson001.demo8.SetterBean$ServiceA"/>
  8. <bean id="setterBean" class="com.javacode2018.lesson001.demo8.SetterBean" autowire="byType" />
  9. </beans>

注意上面setterBean的定义,autowire=”byType”采用了按照类型自动注入的方式,容器启动的时候,会自动取调用SetterBean的setService方法,在容器中查找和这个方法参数类型匹配的bean,将查找的bean通过setService方法注入进去。

来个测试用例,PrimaryTest中加个方法:


  1. @Test
  2. public void setterBean() {
  3. String beanXml = "classpath:/com/javacode2018/lesson001/demo8/setterBean.xml";
  4. ClassPathXmlApplicationContext context = IocUtils.context(beanXml);
  5. }

运行输出:


  1. org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'setterBean' defined in class path resource [com/javacode2018/lesson001/demo8/setterBean.xml]: Unsatisfied dependency expressed through bean property 'service'; nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.javacode2018.lesson001.demo8.SetterBean$IService' available: expected single matching bean but found 2: serviceA,serviceB
  2. at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireByType(AbstractAutowireCapableBeanFactory.java:1526)
  3. at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1406)

错误中重点信息:


  1. org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.javacode2018.lesson001.demo8.SetterBean$IService' available: expected single matching bean but found 2: serviceA,serviceB

容器中去找IService接口对应的bean,期望有一个匹配的,实际上却找到了2个匹配的,不知道如何选择,报错了。

上面2个案例报的异常都是下面这个异常:


  1. org.springframework.beans.factory.NoUniqueBeanDefinitionException

当希望从容器中获取到一个bean对象的时候,容器中却找到了多个匹配的bean,此时spring不知道如何选择了,处于懵逼状态,就会报这个异常。

spring中可以通过bean元素的primary属性来解决这个问题,可以通过这个属性来指定当前bean为主要候选者,当容器查询一个bean的时候,如果容器中有多个候选者匹配的时候,此时spring会返回主要的候选者。

下面我们使用primary来解决上面案例的问题:


  1. package com.javacode2018.lesson001.demo8;
  2. public class PrimaryBean {
  3. public interface IService{} //@1
  4. public static class ServiceA implements IService{} //@2
  5. public static class ServiceB implements IService{} //@3
  6. private IService service;
  7. public void setService(IService service) {
  8. this.service = service;
  9. }
  10. @Override
  11. public String toString() {
  12. return "PrimaryBean{" +
  13. "service=" + service +
  14. '}';
  15. }
  16. }

spring配置文件如下:


  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
  6. <bean id="serviceA" class="com.javacode2018.lesson001.demo8.PrimaryBean$ServiceA" primary="true"/>
  7. <bean id="serviceB" class="com.javacode2018.lesson001.demo8.PrimaryBean$ServiceA"/>
  8. <bean id="setterBean" class="com.javacode2018.lesson001.demo8.PrimaryBean" autowire="byType" />
  9. </beans>

上面配置中我们将serviceA的primary属性置为true了,将其置为主要候选者,容器中如果查找bean的时候,如果有多个匹配的,就以他为主。

我们来个测试用例:


  1. @Test
  2. public void primaryBean() {
  3. String beanXml = "classpath:/com/javacode2018/lesson001/demo8/primaryBean.xml";
  4. ClassPathXmlApplicationContext context = IocUtils.context(beanXml);
  5. PrimaryBean.IService service = context.getBean(PrimaryBean.IService.class); //@1
  6. System.out.println(service);
  7. PrimaryBean primaryBean = context.getBean(PrimaryBean.class); //@2
  8. System.out.println(primaryBean);
  9. }

@1:从容器中查找IService类型匹配的bean,这个接口有2个实现类(ServiceA和Service2),这类在容器中都定义了,但是serviceA为主要的bean,所以这行代码会返回serviceA

@2:从容器中查找PrimaryBean类型的bean,容器中有一个,这个bean按照byType默认注入IService接口匹配的bean,注入的时候如果候选者有多个,以primary=”true”的bean为主来注入,所以此处会注入service2

我们来运行一下,看看和分析的是否一致,运行输出:


  1. com.javacode2018.lesson001.demo8.PrimaryBean$ServiceA@7b227d8d
  2. PrimaryBean{service=com.javacode2018.lesson001.demo8.PrimaryBean$ServiceA@7b227d8d}

上面输出确实都是ServiceA,和我们分析的一致。

还有当候选者中如果有多个bean都将primary置为true,此时spring还是会懵逼的,也会报错,不知道如何选择了。

总结

当从容器中查找一个bean的时候,如果容器中出现多个Bean候选者时,可以通过primary=”true”将当前bean置为首选者,那么查找的时候就会返回主要的候选者,否则将抛出异常。

案例源码


  1. 链接:https://pan.baidu.com/s/1p6rcfKOeWQIVkuhVybzZmQ
  2. 提取码:zr99
<think>嗯,用户想了解关于com.google.mainline.primary.libs.apex的详细信息,这是Android的APEX库。首先,我需要回忆一下APEX是什么。APEX是Android Package EXecutable,用于模块化系统更新,特别是系统级组件。 然后,这个库的具体细节可能包括它的功能,比如提供核心库的更新,比如媒体、网络等。可能还涉及如何集成和使用,比如在项目中依赖这个库。不过用户提到的是com.google.mainline.primary.libs.apex,可能属于Google Mainline项目的一部分,用于通过Google Play更新系统组件。 接下来,用户可能需要知道如何添加依赖项,比如在build.gradle中添加依赖,但需要确认具体的Gradle配置。另外,APEX包的结构、权限、签名等也是关键点。不过可能用户是开发者,需要集成或调试相关库,所以提供使用方法和注意事项比较重要。 还要注意引用来源,比如Android开发者文档或官方资源。可能用户遇到了依赖问题,或者想了解如何更新系统组件,所以相关问题可以涉及APEX的优势、创建自己的APEX包、依赖冲突解决等。</think>关于Android APEX库$com.google.mainline.primary.libs.apex$的技术细节,这是Google Mainline项目的重要组成部分,主要用于通过Google Play商店实现系统组件的模块化更新[^1]。 ### 关键特性 $$ APEX格式 = (压缩文件结构 + 原生二进制文件 + 元数据) $$ 1. **系统级更新**:允许在不更新完整系统镜像的情况下更新核心组件如媒体编解码器、网络协议栈等 2. **安全沙盒**:在受限的权限环境中运行,遵循Android沙盒安全模型 3. **版本控制**:支持多版本共存和回滚机制 ### 典型依赖配置 ```gradle dependencies { implementation 'com.google.android.apex:apex-core:1.0.0' compileOnly 'com.google.mainline:primary.libs.apex:${latest_version}' } ``` ### 注意事项 1. 需要目标设备支持APEX更新机制(Android 10+) 2. 必须使用平台签名密钥进行签名验证 3. 调试时需要启用`adb install --staged`分阶段安装模式[^1]
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值