文章目录
一、Bean的自动装配
- 自动装配是Spring满足bean依赖的一种方式!
- Spring会在上下文中自动寻找,并自动给bean装配属性!
在Spring中有三种装配方式:
- 在xml中显示的配置
- 在java中显示的配置
- 隐式的自动装配bean【🌻重点哦🌻】
1.1 测试案例准备
一个人有两个宠物
(1)创建子工程(spring-04-autowired)
(2)创建对象(Cat.java,Dog.java,People.java)
Cat.java
package com.zql.pojo;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
public class Cat implements Serializable {
public void shout(){
System.out.println("喵喵喵🐈🐈🐈");
}
}
Dog.java
package com.zql.pojo;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
public class Dog implements Serializable {
public void shout(){
System.out.println("汪汪汪🐕🐕🐕");
}
}
People.java
package com.zql.pojo;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
public class People implements Serializable {
private Cat cat;
private Dog dog;
private String name;
public Cat getCat() {
return cat;
}
public void setCat(Cat cat) {
this.cat = cat;
}
public Dog getDog() {
return dog;
}
public void setDog(Dog dog) {
this.dog = dog;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "People{" +
"cat=" + cat +
", dog=" + dog +
", name='" + name + '\'' +
'}';
}
}
(3) 创建beans.xml
applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="cat" class="com.zql.pojo.Cat"></bean>
<bean id="dog" class="com.zql.pojo.Dog"></bean>
<bean id="people" class="com.zql.pojo.People">
<property name="name" value="Daniel🤣🤣"></property>
<property name="cat" ref="cat"></property>
<property name="dog" ref="dog"></property>
</bean>
</beans>
(4)测试类
MyTest.java
import com.zql.pojo.People;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* @Author:Daniel
* @Version 1.0
*/
public class MyTest {
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
People people = context.getBean("people", People.class);
System.out.println(people.getName());
people.getCat().shout();
people.getDog().shout();
}
}
1.2 装配分类
1.2.1 ByName自动装配
<!--ByName:会自动在容器上下文中查找,和自己对象set方法后面的值对应bean id!-->
<bean id="people" class="com.zql.pojo.People" autowire="byName">
<property name="name" value="Daniel🤣🤣"/>
</bean>
1.2.2 ByType自动装配
<bean class="com.zql.pojo.Cat"></bean>
<bean class="com.zql.pojo.Dog"></bean>
<!--ByName:会自动在容器上下文中查找,和自己对象set方法后面的值对应bean id!-->
<!--ByType:会自动在容器上下文中查找,和自己对象属性类型相同的 bean!-->
<bean id="people" class="com.zql.pojo.People" autowire="byType">
<property name="name" value="Daniel🤣🤣"/>
</bean>
总结:
- byname的时候,需要保证所有bean的id唯一,并且这个bean需要和自动注入的属性的set方法的值一致!
- bytype的时候,需要保证所有bean的class唯一,并且这个bean需要和自动注入的属性的类型一致!
1.2.3 使用注解实现自动装配
官网:使用注解实现自动装配
- jdk 1.5 支持的注解,Spring 2.5就支持支持注解了!
The introduction of annotation-based configuration raised the question of whether this approach is “better” than XML.
既然使用注解,就要知道:👇🏾👇🏾
- 导入约束:
context约束
- 配置注解的支持:
context:annotation-config/
<?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
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/>
</beans>
@Autowired
- 直接在属性上使用即可!也可以在set方法上使用!
- 使用Autowired时,我们可以不用编写set方法,前提是你这个自动装配的属性在IOC(Spring)容器中存在,且符合名字byname!
科普:
@Nullable
字段标记了这个注解,说明这个字段为null;
注解 @Autowired
底层源码👇🏾👇🏾
public @interface Autowired {
boolean required() default true;
}
测试代码:
People.java
package com.zql.pojo;
import org.springframework.beans.factory.annotation.Autowired;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
public class People implements Serializable {
//如果显示定义了Autowired的required属性为false,说明这个对象可以为null,否则不允许为空
@Autowired(required = false)
private Cat cat;
@Autowired
private Dog dog;
private String name;
public Cat getCat() {
return cat;
}
public Dog getDog() {
return dog;
}
public String getName() {
return name;
}
@Override
public String toString() {
return "People{" +
"cat=" + cat +
", dog=" + dog +
", name='" + name + '\'' +
'}';
}
}
如果 @Autowired自动装配的环境比较复杂,自动装配无法通过一个注解【 @Autowired】完成的时候,我们可以使用 @Qualifier(value=“xxx”)去配置@Autowired的使用,指定一个唯一的bean对象注入!
public class People implements Serializable {
//如果显示定义了Autowired的required属性为false,说明这个对象可以为null,否则不允许为空
@Autowired(required = false)
private Cat cat;
@Autowired
@Qualifier(value="dog222") //dog222为beans.xml中id的参数
private Dog dog;
private String name;
}
@Resource注解
小结:
public class People implements Serializable {
@Resource(name="cat2")
private Cat cat;
@Resource(name="dog")
private Dog dog;
private String name;
}
@Resource 和 @Autowired的区别
- 都是用来自动装配的,都可以放在属性字段上。
- @Autowired通过byname的方式实现,而且必须要求这个对象存在!【常用】
- @Resource默认通过byname的方式实现,如果找不到名字,则通过byType实现!如果两个都找不到的情况下,就报错!
- 执行顺序不同: @Autowired通过byType的方式实现。 @Resource默认通过byname的方式实现。
二、Spring使用注解开发
2.1 bean
在Spring4之后,要使用注解开发,必须要保证aop的包导入
使用注解需要导入context约束,增加注解的支持!
<?xml version="1.0" encoding="UTF-8"?>
<!--suppress ALL -->
<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"
xmlns:aop="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
https://www.springframework.org/schema/aop/spring-aop.xsd">
<!--开启注解的支持-->
<context:annotation-config/>
</beans>
(1) 创建子工程(spring-05-anno)
(2) User.java
package com.zql.pojo;
import org.springframework.stereotype.Component;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
//组件 相当于: <bean id="user" class="com.zql.pojo.User"></bean>
@Component
public class User implements Serializable {
public String name = "Daniel";
}
(3) applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<!--suppress ALL -->
<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"
xmlns:aop="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
https://www.springframework.org/schema/aop/spring-aop.xsd">
<!--指定要扫描的包,这个包下的注解就会生效-->
<context:component-scan base-package="com.zql.pojo"></context:component-scan>
<!--开启注解的支持-->
<context:annotation-config/>
</beans>
(4) MyTest.java
import com.zql.pojo.User;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* @Author:Daniel
* @Version 1.0
*/
public class MyTest {
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
User user = context.getBean("user", User.class);
System.out.println(user.name);
}
}
- @Component:组件,放在类上,说明这个类被Spring管理了,就是bean!
2.2 属性如何注入
User.java
package com.zql.pojo;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
//组件 相当于: <bean id="user" class="com.zql.pojo.User"></bean>
@Component
public class User implements Serializable {
@Value("Jenny") //相当于<property name="name" value="Jenny"></property>
public String name;
}
2.3 衍生的注解
@Component 有几个衍生注解,我们在web开发中,会按照mvc三层架构分层
- dao【@Repository】
- service【@Service】
- controller【@Controller】
UserDao.java
package com.zql.dao;
import org.springframework.stereotype.Repository;
/**
* @Author:Daniel
* @Version 1.0
*/
@Repository
public class UserDao {
}
UserService.java
package com.zql.service;
import org.springframework.stereotype.Service;
/**
* @Author:Daniel
* @Version 1.0
*/
@Service
public class UserService {
}
UserController.java
package com.zql.controller;
import org.springframework.stereotype.Controller;
/**
* @Author:Daniel
* @Version 1.0
*/
@Controller
public class UserController {
}
applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<!--suppress ALL -->
<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"
xmlns:aop="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
https://www.springframework.org/schema/aop/spring-aop.xsd">
<!--指定要扫描的包,这个包下的注解就会生效-->
<context:component-scan base-package="com.zql"></context:component-scan>
<!--开启注解的支持-->
<context:annotation-config/>
<bean id="user" class="com.zql.pojo.User">
<property name="name" value="Jenny"></property>
</bean>
</beans>
这四个注解功能都是一样的,都是代表将某个注册到Spring中,装配Bean
2.4 自动装配配置
- @Autowired :自动装配通过类型。名字
- 如果Autowired 不能唯一自动装配上属性,则需要通过@Qualifier(value=“xxx”)
- @Nullable 字段标记了这个注解,说明这个字段可以为null
- @Resource:自动装配通过名字。类型
2.5 作用域
package com.zql.pojo;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
import java.io.Serializable;
/**
* @Author:Daniel
* @Version 1.0
*/
//组件 相当于: <bean id="user" class="com.zql.pojo.User"></bean>
@Component
@Scope("singleton") //单例 :singleton 注解开发: prototype
public class User implements Serializable {
public String name;
@Value("Jenny") //相当于<property name="name" value="Jenny"></property>
public void setName(String name) {
this.name = name;
}
}
2.6 小结 (xml与注解)
xml与注解:
- xml更加万能,适用于任何场合!维护简单方便
- 注解 不是自己类使用不了,维护相对复杂!
xml与注解最佳实践:
- xml用来管理bean;
- 注解只负责完成属性的注入;
- 我们在使用的过程中,只需要注意一个问题:必须让注解生效,就需要开启注解的支持
<!--指定要扫描的包,这个包下的注解就会生效-->
<context:component-scan base-package="com.zql"></context:component-scan>
<!--开启注解的支持-->
<context:annotation-config/>
注解说明总结:
- @Autowired :自动装配通过类型。名字
- 如果Autowired 不能唯一自动装配上属性,则需要通过@Qualifier(value=“xxx”)
- @Nullable 字段标记了这个注解,说明这个字段可以为null
- @Resource:自动装配通过名字。类型
- @Component:组件,放在类上,说明这个类被Spring管理了,就是bean!
三、 使用Java的方式配置Spring(拓展)
- 我们现在要完全不使用Spring的xml配置了,全权交给Java来做!
- JavaConfig是Spring的一个子项目,在Spring 4 之后,他成为了一个核心功能!
官网:👉🏾👉🏾 @Configuration
这种配置方式,在后期总结的Springboot中会随处可见!