如何给Spring 容器提供配置元数据/ Spring Bean 定义有几种方式/配置元数据方式有几种?
这里有三种重要的方法给Spring 容器提供配置元数据。
基于XML文件的配置
传统上,配置元数据以简单直观的XML格式提供,如下文件中所示:id是标识单个bean定义的字符串,class定义bean的类型,并使用完全限定的类名,可以选择是否配置属性值。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="/schema/beans"
xmlns:xsi="http:///2001/XMLSchema-instance"
xsi:schemaLocation="/schema/beans
/schema/beans/">
<!-- 空值的user -->
<bean class=""></bean>
<!-- 带值的user -->
<bean class="">
<property name="name" value="张三"></property>
<property name="age" value="18"></property>
</bean>
<!-- 全参构造,使用成员属性索引对应 -->
<bean class="">
<constructor-arg index="0" value="张三"></constructor-arg>
<constructor-arg index="1" value="18"></constructor-arg>
</bean>
<!-- 全参构造,使用成员属性类型对应 -->
<bean class="">
<constructor-arg type="" value="张三"></constructor-arg>
<constructor-arg type="int" value="18"></constructor-arg>
</bean>
</beans>
<property name="name" value="张三"></property>
<constructor-arg index="0" value="张三"></constructor-arg>
XML文件使用ClassPathXmlApplicationContext实例化Spring容器,ApplicationContext是一个维护bean定义以及相互依赖的注册表的高级工厂的接口,通过使用方法T getBean(String name, Class<T> requiredType),可以获取bean的实例,但是理想情况下,应用程序代码永远不要使用它们。
@Test
public void xmlTest(){
ApplicationContext context = new ClassPathXmlApplicationContext("");
User userNoValue = (User)("userNoValue", );
User user = (User)("user", );
User userOnIndex = (User)("userOnIndex", );
User userOnType = (User)("userOnType", );
(userNoValue);
(user);
(userOnIndex);
(userOnType);
}
输出结果:
User(name=null, age=0)
User(name=张三, age=18)
User(name=张三, age=18)
User(name=张三, age=18)
基于Java类的配置
从Spring 3.0开始,Spring JavaConfig项目提供的许多功能成为Spring Framework核心的一部分。因此,可以使用Java而不是XML文件来定义应用程序类外部的bean。基于Java类的配置的核心是带@Configuration(类似于XML中的<beans/>)注解的类和-带@Bean(类似于XML中的<bean/>)注解的方法。@Bean注解标识一个方法,这个方法实例化、配置并且初始化一个由Spring IoC容器进行管理的新对象。@Bean可以对任何使用Spring @Component注解的类中的方法使用,但是最常与@Configuration一起使用。使用@Configuration的类表示其是作为Bean定义的来源,此外,@Configuration类允许通过调用类中其他@Bean方法来定义Bean之间的依赖关系。
@Configuration
public class BeanConfiguration {
@Bean
public Account account(){
return new Account("001001001");
}
@Bean
public User user(Account account){
return new User("张三",18,account);
}
@Bean
public User userNoAccount(){
return new User("张三",18,null);
}
}
通过AnnotationConfigApplicationContext实例化Spring容器,这种通用的ApplicationContext实现方式不仅可以接受@Configuration类作为输入,还可以接受普通@Component类和使用JSR-330元数据注解的类。当@Configuration类作为输入时,@Configuration类和该类中所有@Bean的方法都会被注册为Bean Definition。当@Component类和JSR-330类作为输入时,它们被注册为Bean Definition,并且假定必要时在这些类中使用例如@Autowired或@Inject的DI元数据。
@Test
public void configurationTest(){
ApplicationContext ctx = new AnnotationConfigApplicationContext();
User user = ("user",);
User userNoAccount = ("userNoAccount",);
(user);
(userNoAccount);
}
输出结果:
User(name=张三, age=18, account=Account(number=001001001))
User(name=张三, age=18, account=null)
基于注解的配置
Spring 提供了进一步典型化的注解,包括 `@Component`、`@Service`、`@Repository` 和 `@Controller`。这些注解都是基于 `@Component` 注解的衍生注解,用于更加清晰地标识不同类型的组件。
1. `@Component`:是最通用的注解,用于标识任何被 Spring 管理的组件。当一个类被标记为 `@Component` 时,Spring 会自动扫描并将其实例化为一个 Bean。
2. `@Service`:用于标识服务层(Service)组件。通常用于定义业务逻辑,可以将其看作是 `@Component` 的特定类型。
3. `@Repository`:用于标识持久层(Repository)组件。通常用于与数据库或其他持久化机制交互的组件,例如 DAO(数据访问对象)。
4. `@Controller`:用于标识控制层(Controller)组件。通常用于处理用户请求、调用服务层并返回响应结果的组件。
这些注解的作用是提供更好的语义化和可读性,使得代码更加清晰和易于理解。此外,这些注解也可以与其他注解结合使用,如 `@Autowired`、`@RequestMapping` 等,以实现更复杂的功能。
需要注意的是,这些注解在功能上是相同的,只是为了更好地区分组件的角色和职责而存在。在使用时,可以根据实际情况选择合适的注解来标识组件,以便更好地组织和管理代码。
@Repository
public class JpaMovieFinder implements MovieFinder {
// implementation elided for clarity
}
@Service
public class SimpleMovieLister {
private MovieFinder movieFinder;
public SimpleMovieLister(MovieFinder movieFinder) {
= movieFinder;
}
}