微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

spring之通过注解方式配置Bean二

上一节讲到了基本的基于注解的配置Bean,但是每个Bean之间是没有关联的,现在我们想实现下面的功能

基本目录:

UserController.java

在这里调用UserService中的add方法

package com.gong.spring.beans.annotation.controller;

import org.springframework.stereotype.Controller;

 com.gong.spring.beans.annotation.service.UserService;

@Controller
public class UserController {
    private UserService userService;
    void execute() {
        System.out.println("UserController的execute方法");
        userService.add();
    }
}

UserService.java

在这里调用UserRepository中的save方法

 com.gong.spring.beans.annotation.service;

 org.springframework.stereotype.Service;

 com.gong.spring.beans.annotation.repository.UserRepository;

@Service
 UserService {
     UserRepository userRepository;
     add() {
        System.out.println("UserService中的add方法");
        userRepository.save();
    }
}

UserRepository.java

 com.gong.spring.beans.annotation.repository;

interface UserRepository {
     save();
}

UserRepositoryImpl.java

 org.springframework.stereotype.Repository;

@Repository(value="userRepository")
class UserRepositoryImpl implements UserRepository{

    @Override
     save() {
        // Todo Auto-generated method stub
        System.out.println("UserReposityImpl的save方法");
    }

}

Main.java

 com.gong.spring.beans.annotation;

 org.springframework.context.ApplicationContext;
 org.springframework.context.support.ClasspathXmlApplicationContext;

 com.gong.spring.beans.annotation.controller.UserController; Main {
    static  main(String[] args) {
        1.创建spring的IOC容器对象
        ApplicationContext ctx = new ClasspathXmlApplicationContext("beans-annotation.xml");
        2.从容器中获取Bean实例
        UserController userController = (UserController) ctx.getBean("userController");
        System.out.println(userController);
     userController.execute();
    }
    
}

beans-annotation.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
        
    <!-- 配置springIOC容器扫描的包 -->    
    context:component-scan base-package="com.gong.spring.beans.annotation">
    </context:component-scan>

beans>

按照上节正常的配置,打印下输出结果:

会报空指针异常,是因为我们还没有在UserController中装配userService属性

组件装配:<context:component-scan> 还会自动注册AutowiredAnnotationBeanPostProcessor实例,该实例可以自动装配具有@Autowired、@Resource和@INject注解的属性

使用@Autowired注解自动装配具有类型兼容的单个Bean属性

  • 构造器:普通字段,即使是非public,一切具有参数的方法都可使用@Autowired注解。
  • 认情况下,所有使用@Autowired注解的属性都需要被设置。当spring找不到匹配的bean来装配属性时,会抛出异常。若某一属性不允许被设置,可以设置@Autowired注解的required属性为false。
  • 认情况下,当springIOC容器存在多个类型兼容的Bean时,通过类型的自动装配将无法工作。此时可在@Qualifier注解里提供Bean的名称。spring允许对方法的入参标注@Qualifier以指定注入bean的名称
  • @Autowired注解也可以应用到数据类型的属性上,此时spring将会把所有匹配的bean进行自动装配
  • @Autowired注解也可以应用在集合属性上,此时spring会读取集合的类型信息,然后自动装配给所有与之兼容的bean。
  • @Autowired用在jav.util.Map上时,若该Map的键值为string,那么spring将自动装配与Map值类型兼容的bean,此时bean的名称为键值。

讲了这么多,就是在要用@Autowired让属性自动装配到相应的bean上,即变成:

@Autowired
 UserService userService;
@Autowired
private UserRepository userRepository;

那么,就可以正常运行了,输出

当然,还有另一种方式就是将@Autowired对setter方法进行注解,而不是属性,即:

     UserRepository userRepository;
    @Autowired
     setUserRepository(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

这样也是可以的。

TestObject.java

 org.springframework.stereotype.Component;

@Component
 TestObject {

}

接下来我们在userRepositoryImpl中加入:

 org.springframework.beans.factory.annotation.Autowired;
 org.springframework.stereotype.Repository;

 com.gong.spring.beans.annotation.TestObject;

@Repository(value="userRepository" UserRepository{
    
    @Autowired
    private TestObject testObject;

    @Override
    );
        System.out.println(testObject);
    }

}

程序是可以运行的:

我们将TestObjec.java中的@Component注解去掉:

警告: Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'userController': Unsatisfied dependency expressed through field 'userService'; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'userService': Unsatisfied dependency expressed through method 'setUserRepository' parameter 0; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'userRepository': Unsatisfied dependency expressed through field 'testObject'; nested exception is org.springframework.beans.factory.NoSuchBeanDeFinitionException: No qualifying bean of type 'com.gong.spring.beans.annotation.TestObject' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}

因为我们设置了:

private TestObject testObject;

没有找到该Bean,就会抛出异常。

我们再加上:说明不必要装配该属性

@Autowired(required=false此时即使没有装配上TestObject也不会抛出异常了,即

(我们删除掉了@Autowired(required=false)private TestObject testObject;System.out.println(testObject);)我们再新建一个UserJdbcRepository.java

 org.springframework.stereotype.Repository;

@Repository
class UserJdbcRepository  UserRepository {

    @Override
     Todo Auto-generated method stub
        System.out.println("UserJdbcRepository的save方法");
    }

}

同时删除掉@Repository(value="userRepository")中的value属性,运行会抛出异常:没有一个唯一的bean

因此存在多个bean的实现类时,我们需要指定bean的名字,在调用调用该名字,即:

     userRepository;
    }

或者,我们这么指定要装配的bean的名字:

     UserRepository userRepository;
    @Autowired
    @Qualifier("userJdbcRepository")
     userRepository;
    }

输出

也可以将该注解放入的setter方法中:

    void setUserRepository(@Qualifier("userJdbcRepository") UserRepository userRepository) {
         userRepository;
    }

@Resource和@Inject和@Autowired类似,一般使用@Autowired就足够了。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。

相关推荐