1、什么是组件扫描
指定一个包路径,Spring会自动扫描该包及其子包所有组件类,当发现组件类定义前有特定的注解标记时,就将该组件纳入到Spring容器中,等价于原来的XML配置bean的功能。
2、指定扫描类路径
使用组件扫描,首先需要在XML配置中指定扫描父级package路径,容器会自动去扫描pers.zky包及其子包下的所有组件,并且实例化bean
<!-- 1、开始扫描,指定扫描的包,此处指定的是pers.zky下面所有子包 --> <context:component-scan base-package="pers.zky"></context:component-scan>
3、自动扫描的注解标记
指定扫描的类路径后,并且不是所有的该路径下的组件都会被扫描到Spring容器,只有在组件类定义前有以下标记的,才会扫描到Spring容器。
@Component 通用注解
@Named 通用注解
@Respository 持久化层组件注解
@Service 业务层组件注解
@Controller 控制层组件注解
4、自动扫描注解命名
当一个组件在扫描过程中被检测到时,会生成一个默认的id值,默认id值为小写开头的类名,也可以在注解中自定义id。
package pers.zky.entity; import java.io.Serializable; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; /*@Component//这种注解,扫描时会自动指定其id为首字母小写后的类名,此处即为"book"*/ @Component("b")//指定注解扫描后的id为"b" public class Book implements Serializable{ private int id; private String name; public Book(){ } public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
@Test public void test(){ String conf="applicationContext.xml"; ApplicationContext ac = new ClassPathXmlApplicationContext(conf); /** * Book实体类使用的是注解注入没有指定类的id时 * Spring规定了注解注入经过组件扫描后,其id为类名首字母小写后的名字 */ /*Book book = ac.getBean("book",Book.class);*/ Book book = ac.getBean("b",Book.class); System.out.println(book);//pers.zky.entity.Book@1738a82 System.out.println(book.getId());//0 System.out.println(book.getName());//null
5、指定组件的作用域
通常Spring管理的组件,默认的作用域是"singleton",如果需要其他的作用域也可以使用@Scope注解,只要在注解中提供作用域的名称即可。
package pers.zky.entity; import java.io.Serializable; import org.springframework.context.annotation.Scope; import org.springframework.stereotype.Component; /** * @author Zky * */ @Scope("prototype") @Component("student") public class Student implements Serializable{ private String name; private int id; private Book book; public Student(){ } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getId() { return id; } public void setId(int id) { this.id = id; } public Book getBook() { return book; } }
Student stu = ac.getBean("student",Student.class); Student stu1 = ac.getBean("student",Student.class); System.out.println(stu==stu1);//false,未指定Scope的属性时默认为sington,输出为true
6、指定初始化和销毁方法
@PostConstruct 指定初始化方法
@PreDestroy 指定销毁方法
@PostConstruct//指定初始化方法 public void init(){ System.out.println("student初始化"); } @PreDestroy//指定销毁方法 public void destroy(){ System.out.println("student销毁"); }
7、指定依赖注入关系
具有依赖关系的bean对象,利用下面任意一种注解都可以实现关系的注入。
1)、基本类型的值注入可以使用 @value() 标记
package pers.zky.entity; import java.io.Serializable; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; @Component("b") public class Book implements Serializable{ @Value("10001") private int id; @Value("西游记") private String name; public Book(){ } public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
2)、@Autowired/@Qualifier可以处理构造器注入和Setter注入
@Autowired写在构造器前面,申明需要为其注入bean。
@Qualifier写在参数前面,申明需要注入的bean的id,注入对象单例时,@Qualifier可以省略。
@Autowired写在属性上面,只会执行构造器的赋值语句,其他代码不会执行。
package pers.zky.entity; import java.io.Serializable; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; /** * @author Zky */ @Component("student") public class Student implements Serializable{ @Value("张三") private String name; @Value("1001") private int id; @Autowired private Book book; public Student(){ } public Student(Book book){ System.out.println("before"); this.book = book; System.out.println("after"); } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getId() { return id; } public void setId(int id) { this.id = id; } public Book getBook() { return book; } public void setBook(Book book) { System.out.println("before_getbook"); this.book = book; System.out.println("after_getbook"); } }
@Test public void test(){ String conf="applicationContext.xml"; ApplicationContext ac = new ClassPathXmlApplicationContext(conf); Student stu = ac.getBean("student",Student.class); System.out.println(stu.getBook().getName()); } Console: 西游记
@Autowired写在构造方法或者set方法上时,会执行里面全部的代码。下面是Student类的@Autowired改写到构造器或者set方法上后的输出结果,Test方法和上面一样。
@Autowired public Student(Book book){ System.out.println("before"); this.book = book; System.out.println("after"); } Console: before after 西游记
<pre name="code" class="java">@Autowired public void setBook(Book book){ System.out.println("before_setbook"); this.book = book; System.out.println("after_setbook"); } Console: before_setbook after_setbook 西游记
3)、使用@Resouce注解
只能用在属性和set方法上上。
用在set 方法中 ,执行所有的方法体。
用在属性上只执行方法的赋值。
@Resouce写在构造器上会报语法错误:The annotation @Resource is disallowed for this location
同样使用上面的例子,删掉@Autowired注解,使用@Resouce注解set方法和属性:
@Resource private Book book; Console: 西游记
<pre name="code" class="java">@Resource public void setBook(Book book) { System.out.println("before_setbook"); this.book = book; System.out.println("after_setbook"); } Console: before_setbook after_setbook 西游记
Setter注入推荐使用@Resource,构造器推荐使用@Autowired。
4)、注入spring的表达式
将db.properties至于src文件目录下,其内容如下:
username=scott password=scott jdbc=jdbc:oracle:thin:localhost:1521:orcl driver=oracle.jdbc.OracleDriver
声明properties集合,读取参数:
<util:properties id="jdbc" location="classpath:db.properties"></util:properties>
编写实体类,用spring表达式注入值:
package pers.zky.entity; import java.io.Serializable; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; @Component public class JdbcOracle implements Serializable{ @Value("#{jdbc.username}") private String username; @Value("#{jdbc.password}") private String password; @Value("#{jdbc.url}") private String url; @Value("#{jdbc.driver}") private String driver; public JdbcOracle(){ } public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } public String getUrl() { return url; } public void setUrl(String url) { this.url = url; } public String getDriver() { return driver; } public void setDriver(String driver) { this.driver = driver; } }
编写测试方法,输出结果:
@Test public void test(){ String conf="applicationContext.xml"; ApplicationContext ac = new ClassPathXmlApplicationContext(conf); JdbcOracle jdbcOracle = ac.getBean("jdbcOracle",JdbcOracle.class); System.out.println(jdbcOracle.getUsername()); System.out.println(jdbcOracle.getPassword()); System.out.println(jdbcOracle.getUrl()); System.out.println(jdbcOracle.getDriver()); } Console: scott scott jdbc:oracle:thin:localhost:1521:orcl oracle.jdbc.OracleDriver