Spring核心之控制反转IOC
IOC 概念和原理
什么是IOC
- 控制反转,把对象创建和对象之间的调用过程,交给Spring进行管理
- 使用IOC目的:为了降低耦合度
- 做入门案例就是IOC实现
IOC 底层原理
- xml解析、工厂模式、反射
图解IOC底层原理
IOC 接口(BeanFactory)
IOC思想基于IOC容器完成,IOC容器底层就是对象工厂
Spring提供IOC容器实现两种方式:(两个接口)
ApplicationContext接口有实现类
FileSystemXmlApplicationContext
configLocation:要写上配置文件在系统盘(某个盘)里的路径
ClassPathXmlApplicationContext
configLocation:要写上类路径
IOC操作Bean管理
概念
什么是Bean管理
- Spring 创建对象
- Spring 注入属性
Bean管理操作有两种方式
- 基于xml配置文件方式实现
- 基于注解方式实现
ⅠIOC操作Bean管理-基于xml方式
基于xml方式创建对象
1 2
| <bean id="user" class="com.company.spring5.User"></bean>
|
- 在Spring配置文件中,使用bean标签,标签里面添加对应属性,就可以实现对象创建
- 在bean标签有很多属性,介绍常用的属性
- id属性:唯一标识
- class属性:类全路径(包和类路径)
- 创建对象时候,默认也是执行无参构造方法
基于xml方式注入属性
DI:依赖注入,就是注入属性
DI是IOC一种具体实现,表示依赖注入,注入属性是在创建对象的基础之上进行完成
第一种注入方式:使用set方法进行注入
- 创建类,定义属性和对应的set方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
|
public class Book { private String bname; private String bauthor; public void setBauthor(String bauthor) { this.bauthor = bauthor; } public void setBname(String bname) { this.bname = bname; } }
|
- 在Spring配置文件对象创建,配置属性注入
1 2 3 4 5 6 7 8 9
| <bean id="book" class="com.company.spring5.testdemo.Book">
<property name="bname" value="易筋经"></property> <property name="bauthor" value="达摩老祖"></property> </bean>
|
第二种注入方式:使用有参数构造方法进行注入
- 创建类,定义属性,创建属性对应有参构造方法
1 2 3 4 5 6 7 8 9 10 11 12 13
|
public class Orders {
private String oname; private String address;
public Orders(String oname, String address) { this.oname = oname; this.address = address; } }
|
- 在Spring配置文件中进行配置
1 2 3 4 5
| <bean id="orders" class="com.company.spring5.testdemo.Orders"> <constructor-arg name="oname" value="computer"></constructor-arg> <constructor-arg name="address"value="China"></constructor-arg> </bean>
|
p名称空间注入(了解)
- 使用p名称空间注入,可以简化基于xml配置方式
- 添加p名称空间在配置文件中
1 2 3 4
| <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"></beans>
|
- 进行属性注入,在bean标签里面进行操作
1 2 3
| <bean id="book" class="com.company.spring5.testdemo.Book" p:bname="九阳神功" p:bauthor="无名氏"> </bean>
|
xml注入其他类型属性
字面量(null,特殊符号)
- null值
1 2 3 4
| <property name="address"> <null/> </property>
|
- 属性值包含特殊符号
1 2 3 4 5 6 7
|
<property name="address"> <value><![CDATA[<<南京>>]]></value> </property>
|
注入属性-外部bean
- 创建两个类service类和dao类
- 在service调用dao里面的方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| package com.company.service;
import com.company.dao.UserDao; import com.company.dao.UserDaoImpl;
public class UserService {
private UserDao userDao;
public void setUserDao(UserDao userDao) { this.userDao = userDao; }
public void add(){ System.out.println("service add................."); userDao.update();
} }
|
- 在spring配置文件中进行配置
1 2 3 4 5 6 7 8 9 10
| <bean id="userService" class="com.company.spring5.service.UserService">
<property name="userDao" ref="userDao"></property> </bean> <bean id="UserDao" class="com.company.spring5.dao.UserDaoImpl"></bean>
|
注入属性-内部bean
- 一对多关系,部门和员工
一个部门有多个员工,一个员工属于一个部门,部门是一,员工是多。
- 在实体类之间表示一对多的关系,员工所属的部门,使用对象类型属性进行表示
1 2 3 4 5 6 7 8 9
| public class Dept {
private String dname;
public void setDname(String dname) { this.dname = dname; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| public class Emp { private String name; private String gender;
private Dept dept;
public void setName(String name) { this.name = name; }
public void setGender(String gender) { this.gender = gender; } }
|
- 在Spring配置文件
1 2 3 4 5 6 7 8 9 10 11 12 13
| <bean id="Emp" class="com.company.spring5.bean.Emp">
<property name="name" value="lucy"></property> <property name="gender" value="女"></property>
<property name="dept"> <bean id="dept" class="com.company.spring5.bean.Dept"> <property name="dname" value="保安部"></property> </bean>
</property> </bean>
|
注入属性-级联赋值
- 第一种写法
1 2 3 4 5 6 7 8 9 10 11
| <bean id="Emp" class="com.company.spring5.bean.Emp"> <property name="name" value="lucy"></property> <property name="gender" value="女"></property> <property name="dept" ref="dept"></property> </bean> <bean id="dept" class="com.company.spring5.bean.Dept"> <property name="dname" value="财务部"></property> </bean>
|
第二种写法
在Emp类中添加属性dept的getDept方法
1 2 3 4 5 6 7 8 9 10 11 12
| <bean id="Emp" class="com.company.spring5.bean.Emp"> <property name="name" value="lucy"></property> <property name="gender" value="女"></property> <property name="dept" ref="dept"></property> <property name="dept.dname" value="技术部"></property> </bean> <bean id="dept" class="com.company.spring5.bean.Dept"> <property name="dname" value="财务部"></property> </bean>
|
xml方式注入集合属性
- 注入数组类型属性
- 注入List集合类型属性
- 注入Map集合类型属性
第一步,创建类,定义数组,list,map,set类型属性,生成对应set方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
| public class Stu {
private String[] course;
private List<String> list;
private Map<String,String> map;
private Set<String> set; public void setCourse(String[] course) { this.course = course; }
public void setList(List<String> list) { this.list = list; }
public void setSet(Set<String> set) { this.set = set; }
public void setMap(Map<String, String> map) { this.map = map; } }
|
第二步,在Spring配置文件进行配置
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
| <bean id="stu" class="com.company.spring5.collectiontype.Stu"> <property name="courses"> <array> <value>java课程</value> <value>数据库课程</value> </array> </property> <property name="list"> <list> <value>张三</value> <value>李四</value> </list> </property> <property name="map"> <map> <entry key="JAVA" value="java" ></entry> <entry key="PHP" value="php"></entry> </map> </property> <property name="set"> <set> <value>MySQL</value> <value>redis</value> </set> </property> </bean>
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| <property name="courseList"> <list> <ref bean="course1"></ref> <ref bean="course2"></ref> </list> </property> </bean>
<bean id="course1" class="com.company.spring5.collectiontype.Course"> <property name="cname" value="Spring5框架"></property> </bean> <bean id="course2" class="com.company.spring5.collectiontype.Course"> <property name="cname" value="Mybatis框架"></property> </bean>
|
- 在Spring配置文件中引入名称空间util
1 2 3 4 5 6 7 8
| <?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:util="http://www.springframework.org/schema/util" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/util/spring-beans.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd"> </beans>
|
- 使用util标签完成list集合注入提取
1 2 3 4 5 6 7 8 9 10 11
| <util:list id="bookList"> <value>易筋经</value> <value>九阳神功</value> <value>九阴真经</value> </util:list>
<bean id="book" class="com.company.spring5.collectiontype.Book"> <property name="list" ref="bookList"></property> </bean>
|
FactoryBean
- Spring有两种类型bean,一种普通bean,另一种工厂bean(FactoryBean)
- 普通bean:在配置文件中定义bean类型就是返回类型
- 工厂bean:在配置文件中定义bean类型可以和返回类型不一样
第一步,创建类,让这个类作为工厂bean,实现接口FactoryBean
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| public class MyBean implements FactoryBean<Course> {
@Override public Course getObject() throws Exception { Course course = new Course(); String cou[]={"math,english"}; course.setCname(cou); return course; }
@Override public Class<?> getObjectType() { return null; }
@Override public boolean isSingleton() { return false; } }
|
第二步,配置Spring配置文件
1
| <bean id="myBean" class="com.company.spring5.factorybean.MyBean"></bean>
|
bean 作用域
- 在Spring里面,可以设置bean实例是单实例还是多实例
- 在Spring里面,默认情况下,bean是单实例对象
1 2 3 4 5 6 7 8 9
| @Test public void testCollection2(){ ApplicationContext context=new ClassPathXmlApplicationContext("bean2.xml"); Book book =context.getBean("book", Book.class); Book book1 =context.getBean("book", Book.class); System.out.println(book); System.out.println(book1);
}
|
如何设置单实例还是多实例
- 在Spring配置文件bean标签里面有属性(scope)用于设置单实例还是多实例
- scope属性值
第一个值默认值,singleton,表示单实例对象
第二个值prototype,表示是多实例对象
1 2 3
| <bean id="myBean" class="com.company.spring5.factorybean.MyBean" scope="prototype"></bean> </beans>
|
1 2 3 4 5 6 7 8 9
| @Test public void testCollection2(){ ApplicationContext context=new ClassPathXmlApplicationContext("bean2.xml"); Book book =context.getBean("book", Book.class); Book book1 =context.getBean("book", Book.class); System.out.println(book); System.out.println(book1);
}
|
- singleton表示单实例,prototype表示多实例
- 设置scope值singleton时候,加载spring配置文件就会创建一个单实例对象
- 设置scope值是prototype时候,不是在加载spring配置文件时候创建对象,在调用getBean方法时候创建多实例对象
bean 生命周期
生命周期
bean生命周期
通过构造器创建bean实例(无参数构造)
为bean的属性设置值和对其他bean引用(调用set方法)
调用bean初始化的方法(需要进行配置初始化方法)
bean可以使用了(对象获取到了)
当容器关闭时候,调用bean的销毁的方法(需要进行配置销毁的方法)
演示bean生命周期
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| public class Orders {
public Orders() { System.out.println("第一步 执行无参构造创建bean实例"); }
private String oname;
public void setOname(String oname) { this.oname = oname; System.out.println("第二步 调用set方法设置属性的值"); }
public void initMethod(){ System.out.println("第三步 执行初始化方法"); }
public void destroyMethod(){ System.out.println("第五步 执行销毁方法"); }
}
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @Test public void testCollection4() {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml"); Orders orders = context.getBean("orders", Orders.class); System.out.println("第四步 获取创建bean实例对象"); System.out.println(orders);
context.close();
}
|
1 2 3
| <bean id="orders" class="com.company.spring5.bean.Orders" init-method="initMethod" destroy-method="destroyMethod"> <property name="oname" value="手机"></property> </bean>
|
bean的后置处理器,bean生命周期有七步
- 通过构造器创建bean实例(无参数构造)
- 为bean的属性设置值和对其他bean引用(调用set方法)
- 把bean实例传递bean前置处理器的方法postProcessBeforeInitialization
- 调用bean初始化的方法(需要进行配置初始化方法)
- 把bean实例传递bean后置处理器的方法postProcessAfterInitialization
- bean可以使用了(对象获取到了)
- 当容器关闭时候,调用bean的销毁的方法(需要进行配置销毁的方法)
演示添加后置处理器效果
- 创建一个类,实现接口BeanPostProcessor,创建后处理器
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| public class MyBeanPost implements BeanPostProcessor { @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { System.out.println("在初始化之前执行的方法"); return bean; }
@Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { System.out.println("在初始化之后执行的方法"); return bean; }
}
|
1 2 3 4 5 6 7 8 9 10 11 12
| @Test public void testCollection4() {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml"); Orders orders = context.getBean("orders", Orders.class); System.out.println("第四步 获取创建bean实例对象"); System.out.println(orders);
context.close();
|
1 2 3
| <bean id="MyBeanPost" class="com.company.spring5.bean.MyBeanPost"></bean> </beans>
|
xml 自动装配
什么是自动装配
- 根据指定装配规则(属性名称或者属性类型),Spring自动将匹配的属性值进行注入
演示自动装配过程
1 2 3 4 5 6 7 8 9 10 11
|
<bean id="emp" class="com.company.spring5.autowire.Emp" autowire="byName">
</bean> <bean id="dept" class="com.company.spring5.autowire.Dept"></bean>
|
1 2 3 4
| <bean id="emp" class="com.company.spring5.autowire.Emp" autowire="byType">
</bean> <bean id="dept" class="com.company.spring5.autowire.Dept"></bean>
|
引入外部属性文件
直接配置数据库信息
- 配置德鲁伊连接池
- 引入德鲁伊连接池依赖jar包
1 2 3 4 5 6 7
| <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="driverClassName" value="com.mysql.jdbc.Driver"></property> <property name="url" value="jdbc:mysql://localhost:3306/userDb"></property> <property name="username" value="root"></property> <property name="password" value="root"></property> </bean>
|
- 引入外部属性文件配置数据库连接池
jdbc.properties
- 创建外部属性文件,properteis格式文件,写数据库信息
- 把外部properties属性文件引入到spring配置文件中
1 2 3 4 5 6 7 8
| <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:util="http://www.springframework.org/schema/util" 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/util http://www.springframework.org/schema/util/spring-util.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/util/spring-util.context.xsd">
|
1 2 3 4 5 6 7 8 9
| <context:property-placeholder location="classpath*:jdbc.properties"/> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="driverClassName" value="${prop.driverClass}"></property> <property name="url" value="${prop.url}"></property> <property name="username" value="${prop.userName}"></property> <property name="password" value="${prop.passwd}"></property> </bean>
|
ⅡIOC操作Bean管理-基于注解方式
- 什么是注解
- 注解是代码特殊标记,格式:@注解名称(属性名称=属性值,属性名称=属性值)
- 使用注解,注解作用在类上面,方法上面,属性上面
- 使用注解目的:简化xml配置
- Spring针对Bean管理中创建对象提供注解
- @Component
- @Service
- @Controller
- @repository
- 上面四个注解功能是一样的,都可以用来创建bean实例
基于注解方式实现对象创建
第一步 引入依赖
第二步 开启组件扫描
1 2 3 4
|
<context:component-scan base-package="com.company.spring5"></context:component-scan>
|
第三步创建类,在类上面添加创建对象注解
- 在注解里面value属性值可以省略不写,默认值是类名称,首字母小写
1 2 3 4 5 6 7 8 9 10 11
|
@Component(value = "userService") public class UserService {
public void add(){ System.out.println("service add......"); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
<context:component-scan base-package="com.company" use-default-filters="false"> <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/> </context:component-scan>
<context:component-scan base-package="com.company"> <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> </context:component-scan>
|
基于注解方式实现属性注入
- @AutoWired:根据属性类型自动装配
第一步把service和dao对象创建,在service和dao类添加创建对象注解
第二步在service注入dao对象,在service类添加dao类型属性,在属性上面使用注解
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| @Service public class UserService {
@Autowired private UserDao userDao;
public void add(){ System.out.println("service add......"); userDao.add(); } }
|
- @Qualifier:根据属性名称注入
这个@Qualifier注解的使用,和上面@Autowired一起使用
1 2 3 4 5 6 7 8 9
| @Repository(value = "userDaoImpl1") public class UserDaoImpl implements UserDao{
@Override public void add() { System.out.println("dao add......");
} }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public class UserService {
@Autowired @Qualifier(value = "userDaoImpl1") private UserDao userDao;
public void add(){ System.out.println("service add......"); userDao.add(); } }
|
- @Resource:可以根据类型注入,可以根据名称注入
1 2 3 4
| @Resource(name="userDaoImpl1") private UserDao userDao;
|
Resource是Javax包里面的
- @Value:注入普通类型属性
1 2
| @Value(value = "abc") private String name;
|
完全注解开发
- 创建配置类,替代xml配置文件
1 2 3 4 5
| @Configuration @ComponentScan(basePackages = "com.company") public class SpringConfig { }
|
- 编写测试类
1 2 3 4 5 6 7 8 9
| @Test public void testService2(){
ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class); UserService userService = context.getBean("userService", UserService.class); System.out.println(userService); userService.add(); }
|
参考:https://frxcat.fun/