本篇内容:
- 1、使用注解定义bean
- 2、包扫描注解
- 3、自动装配Bean
Spring成功启动的主要事项:Bean定义信息,Bean实现类,Spring本身。
一、定义Bean
- 采用基于xml的配置,Bean定义信息和Bean实现本省是分离的,采用基于注解的配置文件,bean定义信息通过在bean实现类上标注注解实现。
@Component("userDao")
public class UserDao{}
说明:
- 使用@Component注解对实体进行标注,Spring容器自动将当前类账号成容器管理的Bean。
Spring还提供了3个功能基本和@Component等效的注解分别用于Dao,Service,Controller
- 1、@Repository:用于对dao实现类标注
- 2、@Service:对Service实现类标注
- 3、@Controller用于对Controller实现类标注
注意:
- 可以用@Component代替此3个注解,不统一使用一个注解是为了看出Bean的真实身份。
二、包扫描注解定义的Bean
- Spring的context命名空间提供了通过扫描包以应用注解定义Bean的方式。
1、xml配置包扫描
<context:component-scan base-package=""/>
说明:
- context:component-scan:包扫描标签
- base-package:基础包,Spring扫描基础包所有类,并从类的注解详细中获取Bean的定义信息。
2、扫描特定类可以添加(resource-pattern)属性过滤
<context:component-scan base-package="" resource-pattern="anno/*.class"/>
说明:
- resource-pattern="anno/*.class":Spring仅会扫描基类包annon子包中的类
3、目录下多种匹配
- 如果扫描基础包下需要包含和需要排除的类同时存在可以使用filter
<context:component-scan base-package="" resource-pattern="anno/*.class">
<context:incloud-filter type="regex" expression="com\.smart\.anno.*" />
<context:exclude-filter type="aspectj" expression="com.smart..*Controller+" />
</context:component-scan>
- context:incloud-filter:表示需要包含的目标类
- context:exclude-filter :表示需要排除的目标类
type类型说明:
3.1、annotation:采用目标类是否标注了某个注解进行过滤
- 示例:com.smart.XxxAnnotation
- 说明:所有标注了XxxAnnotation的类,
3.2、assignable:采用目标类首付继承或扩展了某个特定的类进行过滤
- 示例:com.smart.XxxService
- 说明:所有继承或扩展XxxService的类
3.3、aspectj:采用AspectJ表达式进行过滤
- 示例:com.smart..*Service+
- 说明:所有类名以Service结束及继承,或扩展他们的类
3.4、regex:采用正则表达式根据目标类的类名进行过滤
- 示例:com.smart.anno..*
- 说明:所有com.smart.anno类下的包
3.5、custom:采用接口方式过滤
示例:com.smart.XxxTyoeFilter
说明:采用XxxTyoeFilter代码方式实现过滤,前提是必须实现TyoeFilter接口
过滤类型中除了custom类型外,aspectj的过滤表达能力是最强的,可以轻易实现其他类型的表达规则。
4、user-default-filters属性
- 属性的默认值位true:表示默认对标注@Component,@Controller,@Service,@Reposity的Bean进行扫描
三、自动装配Bean
1、使用@Autowired进行自动注入
@Service
public class LoginService{
@Autowired
private LoginDao loginDao;
}
说明
- @Service注解将LoginService标注为一个Bean
- @Autowired注入一个LoginDao实体Bean
- @Autowired默认按照类型匹配方式从容器中查找匹配的Bean,当只有一个匹配的Bean时,Spring将其注入@Autowired标记的变量中
2、使用@Autowired的required属性
- 如果容器中没有和标注变量类型匹配的Bean,Spring容器启动时会抛出异常,如果希望Spring不抛出异常可以添加required属性
@Service
public class LoginService{
@Autowired(required=false)
private LoginDao loginDao;
}
说明
- Autowired的required属性为true
3、使用@Qualifier指定注入Bean的名称
- 如果容器中存在多个匹配的Bean时,采用@Qualifier属性限定Bean的名称
@Service
public class LoginService{
@Autowired
private LoginDao loginDao;
@Autowired
@Qualifier("userDao")
private UserDao userDao;
}
说明:
- 指定UserDao的实例对象在当前当前类注入的bean名称为"userDao"
4、对类方法进行标注
- @Autowired可以对类采用变量和方法的参数进行标注
@Service
public class LoginService{
private LoginDao loginDao;
private UserDao userDao;
//自动将LoginDao传给方法入参
@Autowired
public void setLoginDao(LoginDao loginDao){
this.loginDao = loginDao;
}
//自动将名为userDao的实体传给方法入参
@Autowired
@Qualifier("userDao")
public void setUserDao(UserDao userDao){
this.userDao = userDao;
}
}
- 如果一个方法有多个入参,默认情况下Spring会自动选择匹配入参类型的Bean进行注入
- Spring允许对方法入参标注@Qualifier来指定注入Bean的名称
@Autowired
public void init(@Qualifier("userDao")UserDao userDao,LoginDao loginDao){
this.userdao = userDao;
this.loginDao =loginDao;
}
- 一般Spring中的bean都是单实例对象,一般不需要特别指定Bean实例的名称,来按名称注入
5、对集合类进行标注
- 如果对类中集合类的变量或方法入参进行@Autowired标注,那么Spring会将容器中类型匹配的所有Bean都自动注入进来。
public class ListComponent{
@Autowired(required=false)
private List<Plugin> plugins;
@Autowired
private Map<String,plugin> pluginMaps;
public List<Plugin> getPlugins(){
return plugins;
}
}
说明:
- plugins注入:Spring会将容器中所有类型为Plugin的Bean注入到这个变量中
- pluginMaps:将Plugin类型的Bean注入Map中,key是Bean的名字,value是对应的bean
@Component
@Order(value = 1)
public class OnePlugin implements Plugin{}
说明:
- @Order(value = 1):指定此实体对象的加载顺序,值越小优先被加载。
6、对延迟依赖注入的支持
- 在Spring容器启动时,在Bean上标记了@Lazy及@Autowired注解的属性不会立即注入属性值,而是延迟到调用此属性时才会注入
@Lazy
@Repository
public class Login{}
@Service
public class LoginService {
@Lazy
@Autowired(required=false)
public void setLogin(LoginDao loginDao){}
}
- 对Bean实施延迟依赖注入,@Lazy注解必须同时标注在属性及目标Bean上,缺一不可。