IOC容器
IOC容器:
IOC(概念和原理)
1.什么是IOC
(1)控制反转,把对象创建和对象之间的调用过程,交给Spring管理
(2)使用IOC目的:为了解耦合
(3)做入门案例就是IOC实现
2.IOC底层原理
(1)xml解析,工厂模式,反射
IOC主要用于解决程序间高耦合度
具体原理:
IOC接口(BeanFactory)
1.IOC思想基于IOC容器完成,IOC容器底层就是对象工厂
2.Spring提供IOC容器实现两种方式:(两个接口)
(1)BeanFactory:IOC容器基本实现,是Spring内部的使用接口,不提供给开发人员进行使用
*加载配置文件时不会创建对象,在获取(使用)对象时才去创建对象
(2)ApplicationContext:BeanFactory接口的子接口,提供更多更强大的功能,一般由开发人员进行使用。
*加载配置文件时候就会把配置文件对象进行创建
3.ApplicationContext接口有实现类
FileSystemXmlApplicationContext:是表示绝对路径(文件存放在哪个盘下哪个文件夹位置)
ClassPathXmlApplicationContext:表示工程路径下的文件地址
IOC操作Bean管理(基于XML)
1.什么是Bean管理:Bean管理指两个操作
(1):Spring创建对象
(2):Spring注入属性
2.Bean管理操作有两种方式
(1)基于xml配置文件方式实现
(2)基于注解方式实现
IOC操作Bean管理(基于xml方式)
1.基于xml方式创建对象
(1)在spring配置文件中,使用bean标签,标签里面添加对应属性,就可以实现对象创建
(2)在bean标签有很多属性,介绍常用的属性
*id属性:唯一标识
*class属性:类的全路径(包类路径)
(3)创建对象时候,默认也是执行无参构造方法完成对象创建
2.基于xml方式注入属性
(1)DI:依赖注入,就是注入属性
3.第一种注入方式:使用set方法注入
(1)创建类,定义属性和对应的set的方法
(2)在Spring配置文件配置对象创建,配置属性注入
4.第二种注入方式:使用有参数构造进行注入
(1)创建类,定义属性,创建属性对应的有参构造方法
(2)在Spring配置文件中进行配置
5.p名称空间注入(了解)
【仍然使用的是set注入】
使用p名称空间注入,可以简化基于xml配置方式
第一步:添加p名称空间在配置文件中
第二部:进行属性注入,在bean标签里面进行操作
IOC操作Bean管理(xml注入其他类型属性)
1.字面量
(1)null值
(2)属性值包含特殊符号
方式一:转义字符(<>)
方式二:把带有特殊符号内容写到CDATA
注入属性-外部Bean
(1)创建两个类Servicr和Dao类
userDao
userDaoImpl
userService
(2)在Service调用Dao里面的方法
(3)在Spring配置文件中进行配置
注入属性-内部Bean和级联赋值
(1)一对多:部门和员工
一个部门有多个员工,一个员工属于多个部门
部门是一,员工是多
(2)在实体类中表示一对多的关系,员工表所属部门,使用对象类型属性进行表示
//部门类 public class Dept { private String dname; public void setDname(String dname) { this.dname = dname; } }
//员工类 public class Emp { private String ename; private String gender; //员工属于某一个部门,适用对象形式表示 private Dept dept; public void setDept(Dept dept) { this.dept = dept; } public void setEname(String ename) { this.ename = ename; } public void setGender(String gender) { this.gender = gender; } }
<!--内部bean--> <bean id="emp" class="hrf.spring5.bean.Emp"> <!--设置两个普通属性--> <property name="ename" value="lucky"></property> <property name="gender" value="女"></property> <!--设置对象类型属性--> <property name="dept"> <bean id="dept" class="hrf.spring5.bean.Dept"> <property name="dname" value="安保部"></property> </bean> </property> </bean>
注入属性-级联赋值
前几步与内部bean相同 只有最后Spring配置文件稍有变化
(1)第一种:
<!--级联赋值--> <bean id="emp" class="hrf.spring5.bean.Emp"> <!--设置两个普通属性--> <property name="ename" value="lucky"></property> <property name="gender" value="女"></property> <property name="dept" ref="dept1"></property> <!--级联赋值--> </bean> <bean id="dept1" class="hrf.spring5.bean.Dept"> <property name="dname" value="财务部"></property> </bean>
(2)第二种:
使用此方法需要在Emp类中生成Get方法
<!--级联赋值--> <bean id="emp" class="hrf.spring5.bean.Emp"> <!--设置两个普通属性--> <property name="ename" value="lucky"></property> <property name="gender" value="女"></property> <!--级联赋值--> <property name="dept" ref="dept1"></property> <property name="dept.dname" value="人事部"></property> </bean> <bean id="dept1" class="hrf.spring5.bean.Dept"> <property name="dname" value="财务部"></property> </bean>
IOC操作Bean管理(XML注入集合属性)
1.注入数组类型属性
2.注入List集合类型属性
3.注入Map集合类型属性
(1)创建类,定义数组,List,Map,Set类型属性,生成对象Set方法
import java.util.List; import java.util.Map; import java.util.Set; public class Stu { //1.数组类型属性 private String[] courses; //2 list集合类型属性 private List<String> list; //3 Map集合类型属性 private Map<String,String> maps; //4 set集合类型属性 private Set<String> sets; public void setCourses(String[] courses) { this.courses = courses; } public void setList(List<String> list) { this.list = list; } public void setMaps(Map<String, String> maps) { this.maps = maps; } public void setSets(Set<String> sets) { this.sets = sets; } }
(2)在Spring配置文件中,进行配置
<bean id="stu" class="hrf.spring5.collection.Stu"> <!--数组类型属性注入--> <property name="courses"> <array> <value>语文</value> <value>数学</value> <value>英语</value> </array> </property> <!--List类型属性注入--> <property name="list"> <list> <value>张三</value> <value>李四</value> <value>王五</value> </list> </property> <!--Map类型属性注入--> <property name="maps"> <map> <entry key="JAVA" value="java"></entry> <entry key="PYTHON" value="python"></entry> </map> </property> <!--set类型属性注入--> <property name="sets"> <set> <value>MySql</value> <value>Redis</value> </set> </property> </bean>
编写测试类
@Test public void testCollection(){ ApplicationContext context = new ClassPathXmlApplicationContext("beanl.xml"); Stu stu = context.getBean("stu", Stu.class); stu.test(); }
结果:
在集合中设置对象类型值
(1)
<!--创建多个course对象--> <bean id="course1" class="hrf.spring5.collection.Course"> <property name="cname" value="Spring5框架"></property> </bean> <bean id="course2" class="hrf.spring5.collection.Course"> <property name="cname" value="MyBatis框架"></property> </bean>
(2)
<!--注入list集合类型,值是对象--> <property name="courseList"> <list> <ref bean="course1"></ref> <ref bean="course2"></ref> </list> </property>
结果:
把集合注入部分提取出来
(1)在Spring配置文件中引入名称空间util
<?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/beans/spring-beans.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd ">
(2)使用util标签完成list集合注入提取
<!--1 提取list集合类型属性注入--> <util:list id="bookList"> <value>易筋经</value> <value>九阴真经</value> <value>九阳神功</value> </util:list> <!--2 提取list集合类型属性注入--> <bean id="book" class="hrf.spring5.collection.Book"> <property name="list" ref="bookList"></property> </bean>
IOC操作基于Bean管理(FactoryBean)
(1)Spring有两种类型Bean,一种普通bean,另外一种工厂Bean(FactoryBean)
(2)普通Bean:在Spring配置文件中定义Bean类型就是返回类型
(3) 工厂Bean:在Spring配置文件中定义Bean类型可以和返回类型不一样
第一步 创建类,让这个类作为工厂Bean,实现接口FactoryBean
第二步 实现接口里面的方法,在实现的方法中定义返回的Bean类型
import hrf.spring5.collection.Course; import org.springframework.beans.factory.FactoryBean; public class MyBean implements FactoryBean<Course> { //定义返回Bean @Override public Course getObject() throws Exception { Course course = new Course(); course.setCname("abc"); return course; } @Override public Class<?> getObjectType() { return null; } @Override public boolean isSingleton() { return FactoryBean.super.isSingleton(); } }
测试类:
@Test public void test3(){ ApplicationContext context = new ClassPathXmlApplicationContext("bean3.xml"); Course course = context.getBean("myBean", Course.class); System.out.println(course); }
配置文件:
<bean id="myBean" class="hrf.spring5.factorybean.MyBean"></bean>
IOC操作Bean管理(Bean作用域)
1.在Spring里面,设置创建Bean实例时单实例还是多实例
2.在Spring里面,默认情况下,Bean是单实例对象
输出:
地址相同
3.如何设置单实例还是多实例
(1)在Spring配置文件bean标签里面有属性(scope)用于设置单实例还是多实例
(2)scope属性值:
第一个值 默认值,singleton
第二个值 prototype,表示是多实例对象
结果:
地址不相同
(3)singleton和prototype区别
第一 singleton单实例,prototype多实例
第二 设置scope值是singleton(或者不设置)时候,加载spring配置文件时候,就会创建单实例对象
设置scope值是prototype时候,不是在加载spring配置文件时创建对象,在调用getBean方法时创建多实例 对象
IOC操作Bean管理(bean生命周期)
1.生命周期
(1)从对象创建到销毁的过程
2.bean生命周期
(1)通过构造器创建bean实例(无参数构造)
(2)为bean的属性设置值和对其他bean引用(调用set方法)
(3)调用bean的初始化的方法(需要进行配置)
(4)bean可以使用了(对象获取到了)
(5)当容器关闭的时候,调用bean的销毁方法(需要进行配置销毁的方法)
3.演示bean生命周期
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("第五步 执行销毁的方法 "); } }
<bean id="orders" class="hrf.spring5.bean.Orders" init-method="initMethod" destroy-method="destroyMethod"> <property name="oname " value="手机"></property> </bean>
@Test public void testbean4(){ // ApplicationContext context = // new ClassPathXmlApplicationContext("bean4.xml"); ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml"); Orders orders = context.getBean("orders", Orders.class); System.out.println("第四步 获取创建bean实例对象"); System.out.println(orders); //手动让bean实例销毁 context.close(); }
4.bean的后置处理器,bean的生命周期有七步
(1)通过构造器创建bean实例(无参数构造)
(2)为bean的属性设置值和对其他bean引用(调用set方法)
(3)把bean实例传递bean后置处理器的方法 postProcessBeforeInitialization
(4)调用bean的初始化的方法(需要进行配置)
(5)把bean实例传递bean后置处理器的方法 postProcessAfterInitialization
(6)bean可以使用了(对象获取到了)
(7)当容器关闭的时候,调用bean的销毁方法(需要进行配置销毁的方法)
5.演示添加后置处理器效果
(1)创建类,实现接口BeanPostProcessor,创建后置处理器
import org.springframework.beans.BeansException; import org.springframework.beans.factory.config.BeanPostProcessor; import org.springframework.lang.Nullable; 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; } }
进行配置后置处理器后,会对整个xml页面里的所有bean都有效果
<!--配置后置处理器--> <bean id="myBeanPost" class="hrf.spring5.bean.MyBeanPost"></bean>
IOC操作基于Bean管理(xml自动装配)
1.什么是自动装配
(1)根据指定装配规则(属性名或者属性类型),Spring自动将匹配的属性值进行注入
2.演示自动装配过程
(1)根据属性名称自动注入
<!--实现自动装配 bean标签属性autowire,配置自动装配 autowire属性常用两个值: byName根据属性名称注入(注入值bean的id值和类属性名称一样) byType根据属性类型注入 --> <bean id="emp" class="hrf.spring5.autowire.Emp" autowire="byName"> <!-- <property name="dept" ref="dept"></property>--> </bean> <bean id="dept" class="hrf.spring5.autowire.Dept"></bean>
(2)根据属性类型自动注入
<bean id="emp" class="hrf.spring5.autowire.Emp" autowire="byType"></bean> <bean id="dept" class="hrf.spring5.autowire.Dept"></bean>
IOC操作基于Bean管理(外部属性文件)
1.直接配置数据库信息
(1)配置德鲁伊连接池
(2)引入德鲁伊连接池依赖jar包
<!--直接配置连接池--> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <!-- 获取properties文件内容,根据key获取,使用spring表达式获取 --> <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="123456"></property> </bean>
2.引入外部属性文件配置数据库连接池
(1)创建外部属性文件,properties格式文件,写数据库信息
(2)把外部peoperties属性文件引入到Spring配置文件中
<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" 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/context/spring-context.xsd ">
(3)在Spring配置文件中使用标签引入外部属性文件
value值与外部属性文件中每个value值最前面的key要一致
<!--引入外部属性文件--> <context:property-placeholder location="classpath:jdbc.properties"/> <!--配置连接池--> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <!-- 获取properties文件内容,根据key获取,使用spring表达式获取 --> <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.password}"></property> </bean>
IOC操作基于Bean管理(基于注解)
1.什么是注解
(1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值,属性名称=属性值,属性名称=属性值......)
(2)使用注解,注解作用在类上面,方法上面,属性上面
(3)使用注解的目的:简化xml配置
2.Spring针对Bean管理中(创建对象)提供注解
【四个对象都可以混用,但是分开使用程序更加清晰】
(1)@Component [普通的容器对象,都可以创建对象]
(2)@Service [一般用于业务逻辑层或者Service]
(3)Controller [一般用在web层]
(4)@Repository [一般用在dao层或持久层]
*上面的四个注解功能都是一样的,都可以用来创建bean实例
3.基于注解方式实现对象创建
第一步 引入依赖
第二步 开启组件扫描
<!--开启组件扫描 1 如果需要扫描多个包,所个包使用逗号隔开 2.扫描那几个包的上层目录 base-package="hrf.spring5" --> <context:component-scan base-package="hrf.spring5.dao,hrf.spring5.service"></context:component-scan>
第三步 创建类,在类上面添加创建对象注解
//在注解里面value属性值可以省略不写 //默认值是类名称,首字母小写 //UserService-----------userService @Component(value = "userService") //value与<bean id="" class=""/>中的id作用一样 public class UserService { public void add(){ System.out.println("service add....."); } }
4.开启组件扫描细节配置
<!--开启组件扫描 1 如果需要扫描多个包,所个包使用逗号隔开 2.扫描多个包情况下,可以直接扫描那几个包的上层目录 base-package="hrf.spring5" --> <context:component-scan base-package="hrf.spring5.dao,hrf.spring5.service"></context:component-scan> <!--示例1 use-default-filters="false" 表示现在不使用默认filter,自己配置filter context:include-filter 设置扫描哪些内容 扫描带有Controller的类-->
<context:component-scan base-package="hrf.spring5" use-default-filters="false">
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
<!--示例2 下面配置扫描包内的所有内容 context:exclude-filter: 设置哪些内容不进行扫描 不扫描带有Controller注解的类 -->
<context:component-scan base-package="hrf.spring5">
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> </context:component-scan>
5.基于注解方式实现(属性注入)
(1)@AutoWired 根据属性类型进行自动装配 (2)@Qualifier 根据属性名称进行注入 (3)@Resource 可以根据类型注入,也可以根据名称注入 (4)@Value 注入普通类型属性(1)使用@AutoWired进行演示:根据属性类型进行自动装配
第一步 把service和dao对象创建,在service和dao类添加创建对象注解
第二步 在service注入dao对象,在service类添加dao类型属性,在属性上面使用注解
import hrf.spring5.dao.UserDao; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; //在注解里面value属性值可以省略不写 //默认值是类名称,首字母小写 //UserService-----------userService @Service(value = "userService") //value与<bean id="" class=""/>中的id作用一样 public class UserService { //定义dao类型属性 //不需要添加set方法 //添加注入属性注解 @Autowired private UserDao userDao; public void add(){ System.out.println("service add....."); } }
(2)@Qualifier:根据属性名称进行注入
这个@Qualifier注解的使用,和上面@Autowired一起使用
//定义dao类型属性 //不需要添加set方法 //添加注入属性注解 @Autowired //根据类型进行注入 @Qualifier(value = "userDaoImpl1") //根据名称进行注入 private UserDao userDao;
(3)@Resource: 可以根据类型注入,也可以根据名称注入
//@Resource //根据类型进行注入 @Resource(name="userDaoImpl1") //根据名称进行注入 private UserDao userDao;
6.完全注解开发(不使用配置文件)
【与使用配置文件不同的只有配置类和测试类两部分,其余不变】
(1)创建配置类,替代xml配置文件
import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; @Configuration //表示该类为配置类,替代xml配置文件 @ComponentScan(basePackages = {"hrf.spring5"}) public class SpringConfig { }
(2)编写测试类