【Spring篇】三.springBean自动装配及Resources

本文详细介绍了Spring框架中的自动装配机制,包括No、byName、byType和Constructor四种方式,以及如何通过XML配置实现自动装配。同时,文章还讲解了Spring中的资源加载接口ResourceLoader及其子类,如UrlResource、ClassResource等,用于处理不同类型的资源文件。

1. 自动装配(Autowiring)

类型
1.1 No : 不做任何操作
1.2 byName : 根据属性名自动装配。根据容器中bean名字查找与属性完全一致的bean,将其自动装配。bean中name重复会报错。
定义dao和service
public class TestDao {
    public void say(String word){
        System.out.println("word = [" + word + "]");
    }
}
public class TestService {
    private TestDao testDao;
    // 提供setter方法
    public void setTestDao(TestDao testDao) {
        this.testDao = testDao;
    }
    // 调用dao中的say()
    public void say(String word){
        this.testDao.say(word);
    }
}
创建spring-autowiring.xml 在beans中声明default-autowire=“byName”
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd"
    default-autowire="byName">
    <bean id="testDao" class="com.wpj.autowiring.dao.TestDao"></bean>
    <bean id="testService" class="com.wpj.autowiring.service.TestService"></bean>
</beans>
定义测试类
@Test
public void testByName(){
    ApplicationContext ac = new ClassPathXmlApplicationContext("spring-autowiring.xml");
    TestService testService = ac.getBean("testService",TestService.class);
    testService.say("this is word");
}
1.3 byType : 根据class自动装配。bean中class重复会报错,并指出不能使用该方式进行自动装配。如果没有找到相匹配则什么都不发生。
dao和service不变,xml中bean中声明default-autowire=“byType” 声明dao的bean的name或id可以去掉
<?xml version="1.0" encoding="UTF-8"?>
<beans ...
    default-autowire="byType">
    <bean class="com.wpj.autowiring.dao.TestDao"></bean>
    <bean id="testService" class="com.wpj.autowiring.service.TestService"></bean>
</beans>
1.4 Constructor : 与byType类似,不同于它应用于构造器参数。如果找不到相匹配的bean,则会报错。
在service中定义构造方法 注入dao,xml中bean中声明default-autowire=“constructor”
public class TestService {
    private TestDao testDao;
    public TestService(TestDao testDao) {
        System.out.println("TestService.TestService");
        this.testDao = testDao;
    }
    public void say(String word){
        this.testDao.say(word);
    }

}
<beans ...
    default-autowire="constructor">
    
    <bean class="com.wpj.autowiring.dao.TestDao"></bean>
    <bean id="testService" class="com.wpj.autowiring.service.TestService"></bean>
</beans>

2. Resources

针对于资源文件的统一接口
- UrlResource : URL对应的资源,根据一个URL地址即可构建
- ClassResource : 获取类路径下的资源文件
- FileSystemResource : 获取文件系统里的资源
- ServletContextResource : ServletContext封装的资源,用于访问ServletContext环境下的资源
- InputStreamResource : 针对于输入流封装的资源
- ByteArrayResource : 针对于字节数组封装的资源
ResourceLoader
 对Resource加载的接口
 注入参数的前缀
 	classpath :  从classpath中去加载
 	file : 从文件系统中去加载
 	http : 从URL去加载
 	(none) : 依赖于ApplicationContext
例子
在resources(java项目的src目录下)中创建一个test.txt 随便输点内容
定义TestResources
public class TestResources implements ApplicationContextAware {
    private ApplicationContext applicationContext;
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
    public void resource() throws IOException {
        Resource resource = applicationContext.getResource("classpath:config.txt");
        // Resource resource = applicationContext.getResource("file:F:\\JavaEE1911\\ideaDevelop3\\idea\\day46\\src\\main\\resources\\config.txt");
        // Resource resource = applicationContext.getResource("url:https://docs.spring.io/spring/docs/5.1.3.RELEASE/spring-framework-reference");
        // Resource resource = applicationContext.getResource("config.txt");
        System.out.println(resource.getFilename());
        System.out.println(resource.contentLength());
    }
}
在spring-resource.xml
<bean id="testResources" class="com.wpj.resources.TestResources"></bean>
Test
public class TestResource {
    @Test
    public void test(){
        ApplicationContext ac = new ClassPathXmlApplicationContext("spring-resource.xml");
        TestResources testResource = ac.getBean("testResources",TestResources.class);
        try {
            testResource.resource();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
<think>我们正在解决Spring无法自动装配ResumeServiceImpl类型Bean的问题。根据用户描述,问题出现在使用@Autowired注解注入ResumeServiceImpl时失败。可能的原因及解决方案:1.**包扫描问题**:Spring没有扫描到ResumeServiceImpl所在的包,因此无法创建Bean。-解决方案:确保在配置类(如使用@SpringBootApplication的类)的包或其子包中。如果不是,请使用@ComponentScan注解手动添加扫描路径。2.**缺少注解**:ResumeServiceImpl类上可能没有添加Spring的注解(如@Service,@Component等),导致Spring无法识别。-解决方案:在ResumeServiceImpl类上添加适当的注解,例如:```java@ServicepublicclassResumeServiceImplimplementsResumeService{//...}```3.**多个同类型Bean**:如果存在多个ResumeServiceImpl类型的Bean,Spring无法确定注入哪一个。-解决方案:使用@Qualifier注解指定Bean的名称,或者确保只有一个Bean被定义。4.**依赖冲突**:可能存在版本冲突或依赖缺失。-解决方案:检查pom.xml或build.gradle文件,确保所有Spring相关依赖版本一致且完整。5.**Bean被代理问题**:如果使用了AOP,可能会因为代理导致Bean类型不匹配。-解决方案:尝试使用@Resource注解代替@Autowired,或者调整代理设置。6.**循环依赖**:如果ResumeServiceImpl与其他Bean存在循环依赖,可能导致注入失败。-解决方案:检查循环依赖,尝试使用setter注入或@Lazy注解解决。7.**Bean的作用域问题**:如果Bean的作用域不是Singleton,可能没有被正确创建。-解决方案:检查作用域设置,确保在需要注入的地方可用。8.**测试环境问题**:在测试类中可能没有正确配置。-解决方案:如果是测试环境,确保测试类上有@SpringBootTest注解。参考用户提供的代码片段(虽然与负载均衡相关,但展示了配置类的写法):在配置类中,他们使用@Bean注解创建了RestTemplate的Bean。但是,对于ResumeServiceImpl,通常我们使用@Component或@Service注解,而不是在配置类中显式定义(除非必要)。因此,重点检查ResumeServiceImpl是否在正确的包路径下,并且有@Service等注解。如果以上都检查无误,可以查看具体的错误信息(如NoSuchBeanDefinitionException),根据异常信息进一步排查。示例代码片段(ResumeServiceImpl):```java@ServicepublicclassResumeServiceImplimplementsResumeService{//实现方法}```配置类(确保扫描到):```java@SpringBootApplicationpublicclassMyApplication{publicstaticvoidmain(String[]args){SpringApplication.run(MyApplication.class,args);}}```如果ResumeServiceImpl不在MyApplication所在包及其子包下,需要添加@ComponentScan:```java@SpringBootApplication@ComponentScan(basePackages={"com.example.resume","com.example.main"})publicclassMyApplication{//...}```最后,如果使用XML配置,请确保在XML中配置了Bean或启用了注解扫描。希望这些解决方案能帮助您解决问题。§§相关问题§§1.如何解决Spring中的NoSuchBeanDefinitionException异常?2.Spring中@ComponentScan注解的使用方法?3.如何解决Spring的循环依赖问题?4.@Autowired和@Resource注解有什么区别?</think>### Spring无法自动装配ResumeServiceImpl类型Bean的原因及解决方案 #### 一、常见原因分析 1. **包扫描问题** Spring未扫描到`ResumeServiceImpl`所在的包,导致无法创建Bean。 - 检查启动类是否在顶级包路径下(如`com.example`) - 确认启动类有`@ComponentScan`或`@SpringBootApplication`(包含扫描功能) 2. **Bean定义缺失** - `ResumeServiceImpl`类缺少`@Service`、`@Component`等注解 - 配置文件(XML/Java Config)中未声明Bean 3. **依赖冲突** - 存在多个`ResumeService`实现类导致类型歧义 - 解决方案:使用`@Qualifier("resumeServiceImpl")`明确指定 4. **作用域不匹配** - `ResumeServiceImpl`使用`@Scope("prototype")`但注入点未配置 - 解决方案:配合`@Lookup`注解或调整作用域 5. **循环依赖** `ResumeServiceImpl`与依赖对象相互引用导致初始化失败 - 解决方案:使用setter注入代替构造器注入 #### 二、解决方案 **步骤1:检查注解配置** ```java // 确保实现类有正确注解 @Service // 或 @Component public class ResumeServiceImpl implements ResumeService { // 实现代码 } ``` **步骤2:验证包扫描范围** ```java @SpringBootApplication @ComponentScan(basePackages = { "com.yourpackage.service", // ResumeServiceImpl所在包 "com.yourpackage.main" // 主启动类包 }) public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` **步骤3:处理依赖歧义** ```java @Autowired @Qualifier("resumeServiceImpl") // 明确指定Bean名称 private ResumeService resumeService; ``` **步骤4:检查依赖完整性** - 确保`ResumeServiceImpl`所有依赖项均可注入 - 避免在构造器中调用未初始化的依赖(使用`@PostConstruct`替代) **步骤5:查看日志关键信息** 关注启动日志中的以下错误类型: ``` NoSuchBeanDefinitionException: No qualifying bean of type 'ResumeService' UnsatisfiedDependencyException: Error creating bean with name '...' ``` #### 、进阶排查 1. **Bean名称冲突** 通过`ApplicationContext.getBeanDefinitionNames()`检查所有Bean定义 2. **代理问题** - 若使用AOP/CGLIB代理,尝试添加`@Scope(proxyMode = ScopedProxyMode.INTERFACES)` 3. **模块化应用** 在Spring Boot多模块项目中: ```java @SpringBootApplication(scanBasePackageClasses = { ResumeServiceImpl.class, OtherConfig.class }) ``` 4. **构建工具过滤** 检查Maven/Gradle是否过滤掉`**/impl/*`目录: ```xml <!-- Maven示例 --> <resources> <resource> <directory>src/main/java</directory> <includes> <include>**/*.class</include> <include>**/impl/*</include> <!-- 确保包含 --> </includes> </resource> </resources> ``` #### 四、预防措施 1. 使用构造函数注入(推荐): ```java @Service public class ResumeServiceImpl { private final Dependency dep; // 构造器自动装配 public ResumeServiceImpl(Dependency dep) { this.dep = dep; } } ``` 2. 单元测试验证装配: ```java @SpringBootTest class ResumeServiceTest { @Autowired // 测试能否注入 private ResumeService resumeService; } ``` > **关键提示**:80%的装配问题可通过检查包扫描范围和注解配置解决。若仍存在问题,建议通过`@ConditionalOnProperty`分阶段启用Bean调试[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值