Spring框架—B站学习
1.问题:
- Spring的Ioc可以解决什么问题
- 如何搭建出来一个Spring基于xml的Ioc环境
- 如何实现类之间的依赖,使用Spring的注入来实现依赖关系的
2.程序的耦合:
- 耦合:程序间的依赖关系,包括:类之间的依赖,方法之间的依赖
- 解耦:降低程序间的依赖关系,应该做到:编译期不依赖,运行时才依赖
- 解耦的思路:
- 使用反射创建对象,避免new 对象
- 通过配置文件获取对象创建的全限定类名
3.理解bean和Javabean
- bean 在计算机英语中表示可重用组件
- javabean表示用java语言编写的可重用组件
- javabean > 实体类
4.一个创建bean对象的工厂就是创建我们service和dao对象的,但如何创建:
- 配置文件:配置service和dao,配置的内容:唯一表示 = 全限定类名(key = value)
- 通过读取配置文件中配置的内容,反射创建对象()
配置文件格式可以是xml也可以是properties,properties比xml好解析
5.IOC:
- 导入 springframework依赖后,在依赖里面会出现以下jar包
beans
,context
,core
,expression
包:这四个包是spring
的核心包aop包
:是基于注解开发要用的包由于spring在导坐标的时候不知道你使用xml还是注解,所以把可能用到的jar包都关联进来了jcl包
:把阿帕奇的日志组件集成到这里面来,做成自己的jar包,这是Spring依赖的日志包,如果没有就报错
6.Spring配置文件中 < bean >标签中的id和class代表的意思是
spring配置文件中,集合了spring的ioc容器管理的所有组件
- < bean >标签:把对象的创建交给spring管理
- id:唯一表识
- class:获取创建对象的权限的类名
例子:<?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 http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 把对象的创建交给spring管理 id:唯一表识,class:获取创建对象的权限的类名--> <bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"></bean> <bean id="accountDao" class="com.itheima.dao.impl.AccountDaoImpl"></bean> </beans>
7.BeanFactory:Spring核心容器的顶层接口
-
ApplicationContext是一个接口(Spring的核心容器),以下是他的三个实现类,获取核心容器对象
-
ClassPathXmlApplicationContext:可以加载类路径下的配置文件,要求配置文件必须在类路径下,
不在的加载不了,也创建不了容器了(常用) -
FileSystemXmlApplicationContext:可以加载磁盘任意路径下的配置文件(必须有访问权限)
-
AnnotationConfigApplicationContext:用于读取注解创建容器的
例:
//获取去核心容器对象 ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml"); ApplicationContext ac1 = new FileSystemXmlApplicationContext("//Users//apple//Desktop//bean.xml");
-
2.核心容器两个接口引发的问题:
-
ApplicationContext接口:(单例对象适用)
它在构建核心容器的时候,创建对象采取的策略是采用立即加载的方式。
也就是说,只要一读取完配置文件马上就创建配置文件中的配置对象 -
BeanFactory接口:(多例对象适用)
它在构建核心容器的时候,创建对象采取的策略是采用延迟加载的方式。
也就是说,什么时候根据id获取对象,什么时候才真正的创建对象
Spring对bean的管理细节
1.创建bean的三种方式(遇到在jar包中的某个方法,希望通过spring的容器来获取,就可以采用创建bean的三种方式来实现)
第一种方式: 使用默认构造函数创建:
在spring的配置文件中使用bean标签,配以id和class属性之后,且没有其他属性和标签时。
采用的就是默认构造函数创建bean对象,此时如果类中没有默认构造函数,则对象无法创建。
例:
public class AccountServiceImpl implements IAccountService {
public AccountServiceImpl(){
System.out.println("对象创建了");
}
public void saveAccount(){
System.out.println("service中的saveAccount方法执行了。。。");
}
}
<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"></bean>
第二种方式:使用普通工厂中的方法创建对象(使用某个类中的方法创建对象,并存入spring容器)
factory-bean对应工厂类的id(唯一表示)
factory-method对应工厂类里面需要的哪个方法
例:
public class InstanceFactory {
public IAccountService getAccountService(){
return new AccountServiceImpl();
}
}
<bean id="instanceFactory" class="com.itheima.factory.InstanceFactory"></bean>
<bean id="accountService" factory-bean="instanceFactory" factory-method="getAccountService"></bean>
第三种方式:使用工厂中的静态方法创建对象(使用某个类中的静态方法创建对象,并存入spring容器)
例:
public class StaticFactory {
public static IAccountService getAccountService(){
return new AccountServiceImpl();
}
}
<bean id="accountService" class="com.itheima.factory.StaticFactory"
factory-method="getAccountService"></bean>
2.bean对象的作用范围(Spring的bean对象默认是单例的)
-
bean标签的scope属性:
作用:用于指定bean的作用范围 -
取值: 常用的就是前两个
- singleton:单例的(默认值)
- prototype:多例的
- request:作用于web应用的请求范围
- session:作用于web应用的会话范围
- global-session:作用于集群环境的会话范围(全局会话范围),当不是集群环境时,它就是session
3.bean对象的生命周期
单例对象
:
1.出生
:当容器创建时对象出生
2.活着
:只要容器还在,对象一直活着
3.死亡
:容器销毁,对象消亡
4.总结
:单例对象的生命周期和容器相同多例对象
:
1.出生
:当我们使用对象时spring框架为我们创建
2.活着
:对象只要是在使用过程中就一直活着。
3.死亡
:当对象长时间不用,且没有别的对象引用时,由Java的垃圾回收器回收
Spring依赖注入(构造函数注入,set方法注入,注解)
对象之间的依赖关系,其实就是给对象上的属性赋值!因为对象上有其他对象的变量,因此存在了依赖…依赖注入是注入到spring容器中
依赖注入:Dependency Injection
IOC的作用:
降低程序间的耦合(依赖关系)
依赖关系的管理:
以后都交给spring来维护
什么是依赖关系:
在当前类需要用到其他类的对象,由spring为我们提供,我们只需要在配置文件中说明
依赖关系的维护:
就称之为依赖注入。
依赖注入:
能注入的数据:有三类
1.基本类型和String
2.其他bean类型(在配置文件中或者注解配置过的bean)
3.复杂类型/集合类型
注入的方式:有三种
1.第一种:使用构造函数提供
2.第二种:使用set方法提供
3.第三种:使用注解提供(明天的内容)
1.构造函数注入
构造函数注入:
使用的标签:constructor-arg
标签出现的位置:bean标签的内部
标签中的属性:
1.type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
2.index:用于指定要注入的数据给构造函数中指定索引位置的参数赋值。索引的位置是从0开始
3.name:用于指定给构造函数中指定名称的参数赋值 (常用的)
=============以上三个用于指定给构造函数中哪个参数赋值===============================
4.value:用于提供基本类型和String类型的数据
5.ref:用于指定其他的bean类型数据。它指的就是在spring的Ioc核心容器中出现过的bean对象
优势:
在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功。
弊端:
改变了bean对象的实例化方式,使我们在创建对象时,如果用不到这些数据,也必须提供。
例:
<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl">
<constructor-arg name="name" value="泰斯特"></constructor-arg>
<constructor-arg name="age" value="18"></constructor-arg>
<constructor-arg name="birthday" ref="now"></constructor-arg>
</bean>
<!-- 配置一个日期对象 -->
<bean id="now" class="java.util.Date"></bean>
2.Set方法注入
set方法注入 :(比较常用的方式)
涉及的标签:property
出现的位置:bean标签的内部
标签的属性:
1.name:用于指定注入时所调用的set方法名称
2.value:用于提供基本类型和String类型的数据
3.ref:用于指定其他的bean类型数据。它指的就是在spring的Ioc核心容器中出现过的bean对象
优势:
创建对象时没有明确的限制,可以直接使用默认构造函数
弊端:
如果有某个成员必须有值,则获取对象是有可能set方法没有执行。
例:
<bean id="accountService2" class="com.itheima.service.impl.AccountServiceImpl2">
<property name="name" value="TEST" ></property>
<property name="age" value="21"></property>
<property name="birthday" ref="now"></property>
</bean>
<!-- 配置一个日期对象 -->
<bean id="now" class="java.util.Date"></bean>
3.注入集合数据
复杂类型的注入/集合类型的注入:
1.用于给List结构集合注入的标签:list ,array ,set
2.用于个Map结构集合注入的标签: map ,props
结构相同,标签可以互换
例:
<bean id="accountService3" class="com.itheima.service.impl.AccountServiceImpl3">
<property name="myStrs">
<set>
<value>AAA</value>
<value>BBB</value>
<value>CCC</value>
</set>
</property>
<property name="myList">
<array>
<value>AAA</value>
<value>BBB</value>
<value>CCC</value>
</array>
</property>
<property name="mySet">
<list>
<value>AAA</value>
<value>BBB</value>
<value>CCC</value>
</list>
</property>
<property name="myMap">
<props>
<prop key="testC">ccc</prop>
<prop key="testD">ddd</prop>
</props>
</property>
<property name="myProps">
<map>
<entry key="testA" value="aaa"></entry>
<entry key="testB">
<value>BBB</value>
</entry>
</map>
</property>
</bean>