目录
1.基于注解的 DI概念
1.1指定多个包的三种方式:
1.1.1 使用多个 context:component-scan 指定不同的包路径?
1.1.2? 指定 base-package 的值使用分隔符
1.1.3 . base-package 是指定到父包名
?2.注解类型的使用
?2.1定义 Bean 的注解@Component、@Value
2.2 byType 自动注入@Autowired(默认)
?2.3?byName 自动注入@Autowired 与@Qualifier
2.4JDK 注解@Resource 自动注入?
3. 注解与 XML 的对比
1.基于注解的 DI概念
注解类型:
创建对象的注解:@Component、@Repository、@Service、@Controller。(来自Spring框架) 简单类型的对象属性赋值:@Value。(来自Spring框架) 引用类型的对象属性赋值:@Autowired、@Qualifier。(来自Spring框架) 引用类型的对象属性赋值:@Resource。(来自JDK) 对于 DI 使用注解,将不再需要在 Spring 配置文件中声明 bean 实例。Spring 中使用注解,需要在原有 Spring 运行环境基础上再做一些改变。需要在 Spring 配置文件中配置组件扫描器,用于在指定的基本包中扫描注解。?
1.1指定多个包的三种方式:
1.1.1 使用多个 context:component-scan 指定不同的包路径?
<!--
声明组件扫描器:使用注解必须加上这个语句
component-scan: 组件扫描器,组件是Java对象
属性base-package: 表示项目中的包名
框架会扫描这个包和子包中的所有类,找到类中的所有注解,
遇到注解后,按照注解表示的功能,去创建对象、给属性赋值
-->
<context:component-scan base-package="com.liuhaiyang.ba01" />
<context:component-scan base-package="com.liuhaiyang.ba02" />
1.1.2? 指定 base-package 的值使用分隔符
? ? ? ??分隔符可以使用逗号(,)分号(;)还可以使用空格,不建议使用空格。?
<!-- 可以使用分隔符(;或者,)指定多个包 -->
<context:component-scan base-package="com.liuhaiyang.ba01;com.liuhaiyang.ba02" />
<context:component-scan base-package="com.liuhaiyang.ba03,com.liuhaiyang.ba04" />
1.1.3 . base-package 是指定到父包名
?????????base-package 的值表是基本包,容器启动会扫描包及其子包中的注解,当然也会扫描到子包下级的子包。所以 base-package 可以指定一个父包就可以。
<!-- 也可以指定父包 -->
<context:component-scan base-package="com.liuhaiyang" />
?2.注解类型的使用
?2.1定义 Bean 的注解@Component、@Value
@Component: 表示创建对象,对象放到容器中,作用是 <bean> ? ? ? ? ? 属性: value,表示对象名称,也就是 <bean> 标签中的 id 属性值 ? ? ? ? ? 位置: 在类的上面,表示创建此类的对象
与@Component功能相同的创建对象的注解:
????????@Repository : 放在dao接口的实现类上面,能访问数据库 ????????@Service : 放在service的实现类上面,具有事务的功能 ????????@Controller : 放在controller的实现类上面,能够接收请求,把请求的处理结果显示给用户。
????????@Repository,@Service,@Controller 是对@Component 注解的细化,标注不同层的对象。即持久层对象,业务层对象,控制层对象
简单类型的属性赋值:@Value ? ? ?? 属性: value,表示属性值,可以省略 ? ? ?? 位置:1) 在属性定义的上面,无需set方法,推荐使用 ? ? ? ? ? ?? ? ? ?2) 在set方法上面
实体类:
package com.liuhaiyang.bao01;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
/**
* @Component: 表示创建对象,对象放到容器中,作用是<bean>
* 属性:value,表示对象名称,也就是bean的id属性
* 位置:在类的上面,表示创建此类的对象。
*
* @Component(value =”mystudent“)等同于
* <bean id="myStudent" clas="com.liuhaiyang.bao01.student">
*
* @Component 功能相同的创建对象的注解
* 1.@Repository:放到dao接口的实现类上面,表示创建dao对象,持久层对象,能访问数据库
* 2.@Service: 放在业务层接口的实现类上面,表示创建业务对象,业务层对象有事务的功能
* 3.Controller: 放在控制器类的上面,表示创建控制器对象。属于表示层对象。
* 控制器对象能接受请求,把请求的处理结果显示给用户。
*
*/
//@Component(value = "myStudent")
//默认名称:类名首字母小写 @Component
//可以省略value
@Component("myStudent")
public class Student {
//使用外部属性文件中的数据,语法@value("${key}")
@Value("张三")
private String name;
@Value("20")
private int age;
public Student(){
System.out.println("调用了无参构造!!");
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
配置文件
<?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 https://www.springframework.org/schema/context/spring-context.xsd">
<!-- 声明组件扫描器:使用注解必须要加入这句话
component-scan:翻译过来是组件扫描器,组件是Java对象。
属性:base-package 注解在你的项目中的包名。
框架会扫描这个包和子包中的所有类,找到类中的所有注解。
遇到注解后,按照注解表示的功能,去创建对象,给属性赋值。
-->
<context:component-scan base-package="com.liuhaiyang.bao01"/>
</beans>
测试类:
@Test
public void mytest01(){
String confed="applicationContext.xml";
ApplicationContext cxt=new ClassPathXmlApplicationContext(confed);
Student student=(Student) cxt.getBean("myStudent");
System.out.println(student);
}
结果截图:

2.2 byType 自动注入@Autowired(默认)
????????需要在引用属性上使用注解@Autowired,该注解默认使用按类型自动装配 Bean 的方式。使用该注解完成属性注入时,类中无需 setter。当然,若属性有 setter,则也可将其加到 setter 上。
属性:required:boolean类型的属性,默认为true ? ? ? ? ? ? true:spring在创建容器对象时,会检查引用类型是否赋值成功,如果赋值失败,终止程序运行并报错。 ? ? ? ? ? ? false:如果引用类型赋值失败,程序正常执行不会报错,此时引用类型的值为null。
位置:1)在属性定义的上面,无需set方法,推荐使用 ? ? ? ? ? ?2)在set方法上面 实体类:
package com.liuhaiyang.bao03;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component("myschool")
public class School {
@Value("厕所蹲大学")
private String name;
@Value("北京丰台区")
private String bname;
public void setName(String name) {
this.name = name;
}
public void setBname(String bname) {
this.bname = bname;
}
@Override
public String toString() {
return "School{" +
"name='" + name + '\'' +
", bname='" + bname + '\'' +
'}';
}
}
package com.liuhaiyang.bao03;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component("myStudent")
public class Student {
@Value("李四")
private String name;
@Value(value = "23")
private int age;
/**
* 引用类型 @Autowired: spring框架提供的,给引用类型赋值的,使用自动注入原理。
* 支持byName,byType。默认是byType
* 位置:1.在属性定义的上面,无需set方法,推荐使用
* 2.在set方法上面
* 属性:required:boolean类型的属性,默认是true spring在启动的时候创建容器对象。
* 会检查引用类型是否赋值成功。当时true时,终止程序,并报错。 当为false:引用类型赋值失效,
* 程序正常执行,不报错引用类型值为null
*
*byName自动注入: 1.@Autowired:给引用类型赋值
* 2.@Qualifer(value="bean的id"):从容器中找到指定名称的对象,把这个对象赋值给引用类型。
*/
@Autowired()
private School school;
public void setName(String name) {
this.name = name;
}
public void setAge(int age) {
this.age = age;
}
public Student(){
System.out.println("调用了无参构造!!");
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
", school=" + school +
'}';
}
}
配置文件
<?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 https://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="com.liuhaiyang.bao03"/>
</beans>
测试类
@Test
public void mytest01(){
String confed="applicationContext.xml";
ApplicationContext cxt=new ClassPathXmlApplicationContext(confed);
Student student=(Student) cxt.getBean("myStudent");
System.out.println(student);
}
结果截图:

?2.3?byName 自动注入@Autowired 与@Qualifier
?????????spring框架提供的,给引用类型赋值的,使用自动注入原理。支持byName、byType,默认是byType
????????需要在引用属性上联合使用注解@Autowired 与@Qualifier。@Qualifier的 value 属性用于指定要匹配的 Bean 的 id 值。类中无需 set 方法,也可加到set 方法上。
以为代码上面代码基本一致,所以写不一样的
Student实体类:
package com.liuhaiyang.bao03;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component("myStudent")
public class Student {
@Value("李四")
private String name;
@Value(value = "23")
private int age;
/**
* 引用类型 @Autowired: spring框架提供的,给引用类型赋值的,使用自动注入原理。
* 支持byName,byType。默认是byType
* 位置:1.在属性定义的上面,无需set方法,推荐使用
* 2.在set方法上面
* 属性:required:boolean类型的属性,默认是true spring在启动的时候创建容器对象。
* 会检查引用类型是否赋值成功。当时true时,终止程序,并报错。 当为false:引用类型赋值失效,
* 程序正常执行,不报错引用类型值为null
*
*byName自动注入: 1.@Autowired:给引用类型赋值
* 2.@Qualifer(value="bean的id"):从容器中找到指定名称的对象,把这个对象赋值给引用类型。
*/
@Autowired(required = false)
@Qualifier("myschoolAAAA") //无先后顺序之分
private School school;
public void setName(String name) {
this.name = name;
}
public void setAge(int age) {
this.age = age;
}
public Student(){
System.out.println("调用了无参构造!!");
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
", school=" + school +
'}';
}
}
其他代码是一样的。就不写了,掌握即可。
2.4JDK 注解@Resource 自动注入?
引用类型:@Resource,来自jdk中,给引用类型赋值,支持byName、byType,默认是byName。Spring框架支持这个注解。
位置:1) 在属性定义的上面,无需set方法,推荐使用 ? ? ? ? ? 2) 在set方法上面
@Resource:先使用byName赋值,如果赋值失败,再使用byType @Resource(name="mySchool"):表示只使用byName 如果使用jdk1.8,则带有@Resource注解;如果高于jdk1.8,则没有这个注解。如需使用,需要在pom.xml文件中加入依赖
<dependency>
<groupId>javax.annotation</groupId>
<artifactId>javax.annotation-api</artifactId>
<version>1.3.2</version>
</dependency>
实体类:
package com.liuhaiyang.bao04;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component("myschool")
public class School {
@Value("厕所蹲大学")
private String name;
@Value("tianj丰台区")
private String bname;
public void setName(String name) {
this.name = name;
}
public void setBname(String bname) {
this.bname = bname;
}
@Override
public String toString() {
return "School{" +
"name='" + name + '\'' +
", bname='" + bname + '\'' +
'}';
}
}
package com.liuhaiyang.bao04;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Repository;
import javax.annotation.Resource;
@Component("myStudent")
public class Student {
@Value("李四")
private String name;
@Value(value = "23")
private int age;
/**引用类型
* @Resource: 来自jdk中,给引用类型赋值的,支持byName,byType。默认是byName。Spring支持这个注解的使用
* 位置:1.在属性上面定义。无需set方法,推荐
* 2.在set方法的上面使用
* 注意:使用jdk1.8带有@Resource注解,高于jdk1.8则没有这个注解,需要加入一个依赖
*/
//先使用byName方式赋值(就是查找myschool),如果失败了,在使用byType(就是查找School类)
@Resource(name = "myschool")
private School school;
public void setName(String name) {
this.name = name;
}
public void setAge(int age) {
this.age = age;
}
public Student(){
System.out.println("调用了无参构造!!");
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
", school=" + school +
'}';
}
}
配置文件
<?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 https://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="com.liuhaiyang.bao04"/>
</beans>
测试类
@Test
public void mytest03(){
String confed="applicationContext.xml";
ApplicationContext cxt=new ClassPathXmlApplicationContext(confed);
Student student=(Student) cxt.getBean("myStudent");
System.out.println(student);
}
结果截图:

3. 注解与 XML 的对比
注解方式
优点:方便、直观、高效(代码少,没有配置文件的书写那么复杂)?。
缺点:以硬编码的方式写入到 Java 代码中,修改是需要重新编译代码的。
XML方式
优点:1.配置和代码是相分离的。
? ? ? ? ? ?2.在xml中做修改,无需编译代码,只需重启服务器即可将新的配置加载。
缺点:编写麻烦,效率低,如果面对大型项目就显得过于复杂。
|