基于注解实现控制反转
声明Bean的注解 @Component
在类上添加注解@Component表示该类创建对象的权限交给Spring容器。注解的value属性用于指定bean的id值,value可以省略。
@Component 则不指定 value 属性,bean 的 id 是类名的首字母小写。
@Component(value = "teamDao")
public class TeamDao {
public void add(){
System.out.println("TeamDao的add方法被调用");
}
public TeamDao() {
System.out.println("TeamDao的无参构造方法被调用");
}
}
除此之外,Spring中还提供了其他3个用于创建对象的注解:
@Repository : 用于dao实现类的的注解
@Service: 用户service实现类的注解
@Controller: 用于controller实现类的注解
这三个注解与@Component 都可以创建对象,但这三个注解还有其他的含义,@Service创建业务层对象,业务层对象可以加入事务功能,@Controller 注解创建的对象可以作为处理器接收用户的请求。
包扫描
需要在 Spring 配置文件中配置组件扫描器,用于在指定的基本包中扫描注解。如果没有报扫描,添加的创建对象的注解不生效。
扫描方式1
使用多个context:component-scan指定不同的包路径
<!--context:component-scan表示告知spring 要扫描的包
这些包以及子包当中的类上如果添加了@Component注解,这些添加了注解的类就交给了spring容器创建对象
注意:在beans标签中添加 xmlns:context="http://www.springframework.org/schema/context"
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
-->
<!--多个包的扫描: 方式1 :使用多个<context:component-scan 表明-->
<context:component-scan base-package="com.kkb.dao"></context:component-scan>
<context:component-scan base-package="com.kkb.service"></context:component-scan>
<context:component-scan base-package="com.kkb.controller"></context:component-scan>
扫描方式2
指定 base-package的值使用分隔符,分隔符可以使用逗号(,)分号(;)还可以使用空格,不建议使用空格。
<!--多个包的扫描: 方式2 : base-package中直接声明要扫描的多个包 ,多个值用逗号,分号或者空格分割,但是空格不推荐-->
<context:component-scan base-package="com.kkb.dao,com.kkb.service,com.kkb.controller"></context:component-scan>
扫描方式3
base-package 的值表是基本包,容器启动会扫描包及其子包中的注解,当然也会扫描到子包下级的子包。所以 base-package 可以指定一个父包就可以。但不建议使用顶级的父包,扫描的路径比较多,导致容器启动时间变慢。指定到目标包和合适的。也就是注解所在包全路径。
<!--多个包的扫描: 方式3: base-package中直接声明要扫描的多个包的父包-->
<context:component-scan base-package="com.kkb"></context:component-scan>
属性注入@Value
需要在属性上使用注解@Value,该注解的 value 属性用于指定要注入的值。使用该注解完成属性注入时,类中无需 setter。当然,若属性有 setter,则也可将其加到 setter 上。
public class Team {
private String id;
@Value("湖人队")
private String name;
@Value("洛杉矶湖人队")
private String introduction;
@Value("0011")
public void setId(String id) {
this.id = id;
}
public void setName(String name) {
this.name = name;
}
public void setIntroduction(String introduction) {
this.introduction = introduction;
}
}
byType自动注入@Autowired
需要在引用属性上使用注解@Autowired,该注解默认使用按类型自动装配 Bean 的方式。使用该注解完成属性注入时,类中无需 setter。当然,若属性有 setter,则也可将其加到 setter 上。
@Autowired 还有一个属性 required,默认值为 true,表示当匹配失败后,会终止程序运行。若将其值设置为 false,则匹配失败,将被忽略,未匹配的属性值为 null。
byName自动注入@Autowired和@Qualifier
需要在引用属性上联合使用注解@Autowired 与@Qualifier。@Qualifier 的 value 属性用于指定要匹配的 Bean 的 id 值。类中无需 set 方法,也可加到 set 方法上。
自动注入@Resource
Spring提供了对 jdk中@Resource注解的支持。@Resource 注解既可以按名称匹配Bean,也可以按类型匹配 Bean。默认是按名称注入。使用该注解,要求 JDK 必须是 6 及以上版本。@Resource 可在属性
上,也可在 set 方法上。
@Controller
public class TeamController {
@Resource(name="teamService1"/*,type = TeamService.class*/) //JDK版本高于1.6 自动装配默认按照名称 但是如果名称没有相符的,就按照类型自动装配
private TeamService teamService;
public void add(){
teamService.add();
System.out.println("TeamController---- add ----");
}
public TeamController() {
System.out.println("TeamController ---- 默认的构造放方法");
}
}
byType注入引用类型属性
@Resource 注解若不带任何参数,采用默认按名称的方式注入,按名称不能注入 bean,则会按照类型进行 Bean 的匹配注入。
byName注入引用类型属性
@Resource 注解指定其 name 属性,则 name 的值即为按照名称进行匹配的 Bean 的 id。