Spring bean 的装配方式总的来说有以下3种:基于xml配置文件的装配、基于注解的装配、基于命名空间的装配。三种装配方式各有优缺点,在实际的开发工作中,根据“约定优于配置”的信条,基于注解的装配方式比较受欢迎。下面介绍各种 装配方式
1、基于配置文件的装配方式
直接上配置文件:
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">
基于配置文件的装配方式如上,没什么需要细讲的东西,最后通过
ApplicationContext ctx =
new ClassPathXmlApplicationContext("classpath:spring-cfg.xml");
获取到ApplicationContext接口的实现类对象。
2、基于注解的装配方式
基于注解的装配方式需要一个注解配置类(和要装配的bean在同一个包路径下或者子包下)
这里我们新建一个配置类AnotationConfig,代码如下:
@ComponentScan(basePackages= {"spring.chapter10.loadbeanbyanotation"}, basePackageClasses= {Role.class})
public class AnotationConfig {
}
可以看到这个类其实并没有任何实际代码,起作用的是上面加粗的注解部分
basePackages指明要装载的bean的包路径或者父包路径,可以指定多个路径(要注意bean id是否重复)
basePackageClasses指定要装载的Bean(不指定默认装载所有的Bean)
要装载的Bean需要用@component注解
//也可简写为@Component("role")
@Component(value = "role")
public class Role {
@Value("administrator")
private String name;
@Value("default")
private String note;
//Spring会把字符串类型转换为Long型
@Value("1")
private Long id;
/**getter and setter*/
}
获取ApplicationContext接口实现对象:
ApplicationContext ctx =
new AnnotationConfigApplicationContext(AnotationConfig.class);
上述的Bean的依赖都是基本数据类型比如 @Value("1") private Long id;那如果依赖的是自定义的类怎么办呢?
这时候就需要@Autowired自动装配注解
@Autowired自动装配注解又分为根据类型自动装配、根据依赖的bean id自动装配、根据优先级自动装配。
默认的装配方式是根据类型自动装配,代码如下:
public class RoleService{
@Autowired
private Role role = null;
/**getter and setter*/
}
但是这里存在一个问题,如果Role是一个接口,而它的实现类不止一个呢,那么spring初始化的时候会抛出异常
这时候可以通过@Primary注解来指定优先的Bean依赖注入
@Component("roleImpl1")
@Primary
public class RoleImpl1 implements Role{
}
@Component("roleImpl2")
public class RoleImpl2 implements Role{
}
那么这时候,Spring会优先将bean roleImpl1注入到RoleService中
除此之外还可以使用@Qualifier注解来根据bean id自动装配
public class RoleService{
@Autowired
@Qualifier("roleImpl2")
private Role role = null;
}
这时候就会使用roleImpl2注入到RoleService中
三种自动装配方式的优先级: 根据bean id 也就是@Qualifier > @Primary > byType也就是默认的根据类型装配
@Autowired还可以用于构造方法参数自动装配
public class RoleService{
private Role role = null;
public RoleService(@Autowired Role role){
this.role = role;
}
3、装配的混合使用
各种装配方式各有优缺点,在实际的开发工作中,工程内的bean通常采用注解进行配置。
第三方的jar包里的bean通常采用xml文件进行配置,这样可以不考虑第三方项目的实现细节。
创建RoleDaoImpl代码如下:
@Component("roleDaoImpl")
public class RoleDaoImpl implements RoleDao{
@Autowired
private DataSource dataSource = null;
public Role getRoleById(Long id) {
Connection connection = null;
PreparedStatement ps = null;
ResultSet rs = null;
String sql = "SELECT * FROM t_role WHERE id = ?";
try {
connection = dataSource.getConnection();
ps = connection.prepareStatement(sql);
ps.setLong(1, id);
rs = ps.executeQuery();
while(rs.next()) {
Role role = new Role();
role.setId(id);
role.setName(rs.getString("name"));
role.setNote(rs.getString("note"));
return role;
}
} catch (SQLException e) {
e.printStackTrace();
}finally {
if(connection != null) {
try {
connection.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
}
return null;
}
}
注解信息配置类AnotationConfig:
@ImportResource("classpath:spring-cfg-chapter10.xml") //可以在注解配置类中引入xml配置,可以通过多个@importResource注解导入多个配置文件
但是spring目前还不支持在xml配置中引入注解类但是可以通过扫包的方式
/**
*
* @author John
* 通过这个配置类,Spring会扫描同包或者子包下的所有注解Component的Bean
*可以指定扫描的包路径和需要扫描的类
*/
@ComponentScan(basePackages= {"spring.chapter10.loadbeanbyanotation"}, basePackageClasses= {Role.class})
public class AnotationConfig {
}
另外也可以根据业务模块产生多个xml配置文件,然后在一个总的配置文件中import它们,如下: