一、Spring的核心 IoC(基于XML)
1、IoC容器
(1)BeanFactory容器创建对象:
//使用BeanFactory
@Test
void testBeanFactory() throws Exception {
Resource resource = new ClassPathResource("com/shan/container/container.xml");
BeanFactory factory = new XmlBeanFactory(resource);
Person person = factory.getBean("person", Person.class);
}
(2)BeanFactory的子接口 ApplicationContext容器创建对象:
//使用ApplicationContext
@Test
void testApplicationContext() throws Exception {
//习惯将上下文对象命名为ctx
ApplicationContext ctx = new ClassPathXmlApplicationContext("com/shan/container/container.xml");
Person person = ctx.getBean("person", Person.class);
}
(3)bean创建时机(ApplicationContext 和 BeanFactory 创建对象的区别 ):
BeanFactory 在创建Spring容器的时候,并不会立马创建容器中管理的Bean对象,需要等到获取某一个 bean 的时候才会创建该 bean--延迟初始化。(懒加载)
ApplicationContext 在启动 Spring 容器的时候就会创建所有的 bean(
在 Web 应用使用Application
)
□ 当然可以通过属性设置懒加载:
对于某个bean对象,在该bean元素上通过 属性lazy-init 进行设置; 对所有bean对象,在beans上通过 属性default-lazy-init进行设置。
2、bean实例化方式(4种):
★ ① 构造器实例化(bean 中有 无参数构造器
),标准、常用
。
② 静态工厂方法实例化:解决系统遗留问题。
③ 实例工厂方法实例化:解决系统遗留问题。
★ ④ 实现 FactoryBean 接口实例化:是第三种方式实例工厂的变种。 如集成 MyBatis 框架使用:org.mybatis.spring.SqlSessionFactoryBean
□ 方式2(工厂类中有一个创建bean对象的静态方法):在xml的配置:
<bean id="" class="工厂类的全限定名" factory-method="工厂类中创建对象的静态方法名"/>
□ 方式3(工厂类中有一个创建bean对象的方法):在xml的配置:
<bean id="factory3" class="工厂类的全限定名"/>
<bean id="create3" factory-bean="factory3" factory-method="工厂类中创建对象的方法名"/>
□ 方式4(实现 FactoryBean 接口实例化的方法):在xml的配置:(在方法3中,配置了两个bean,咱的思路是:将其中一个bean "固定化",
做法:将工厂实例化对象的方法规定为都叫 getObject 方法,因为咱定义的工厂类实现了FactoryBean接口(这是该接口的规范))
public class DogFactory implements FactoryBean<Dog>{
@Override
public Dog getObject() throws Exception {
Dog dog = new Dog();
return dog;
}
@Override
public Class<?> getObjectType() {
return Dog.class;
}
}
<!-- 实现 FactoryBean 接口实例化:实例工厂变种, 如集成 MyBatis 框架使用 -->
<bean id="dog" class="com.shan._04_factory_bean.DogFactory"/>
/* 使用spring的测试框架 */
@SpringJUnitConfig
public class App {
@Autowired
private Person person;
@Autowired
private BeanFactory factory;
@Autowired
private ApplicationContext ctx;
@Autowired
private Dog dog;
@Test
void testBeanFactory() throws Exception {
System.out.println(person);
System.out.println(dog);
}
}
3、bean作用域scope(常用单例)
-
在 Spring 容器中是指其创建的 Bean 对象相对于其他 Bean 对象的请求可见范围,定于语法格式:
<bean id="" class="" scope="作用域"/>
单例和多例: singleton: 单例(默认的作用域) prototype: 多例
在web应用中(request、session、application)
globalSession: 一般用于 Porlet 应用环境 , 分布式系统存在全局 session 概念(单点登录)
websocket:将一个bean定义定义到WebSocket的生命周期
缺省情况下是单例 singleton
4、bean初始化和销毁
- 属性init-method="该类中初始化方法名" 和 属性destroy-method="该类中销毁方法名"
- 没有使用spring的测试框架的话,就不能正常关闭IoC容器,即销毁bean对象了(可以手动关闭)
<bean id="cat" class="com.shan.lifecycle.Cat" init-method="init" destroy-method="close"/>
■ bean 的生命周期:bean 从出生到消亡的整个过程。
BeanFactory:延迟初始化特点
ApplicationContext:在启动Spring容器的时候,就会去创建bean对象
5、bean的生命周期
①执行Bean 构造器 ②为Bean注入属性 ③调用Bean元素的init-method 进行初始化 ④获取Bean对象,调用Bean对象的某个方法 ⑤调用Bean元素的destroy-method 进行销毁对象 ⑥销毁Spring容器
二、Spring的 DI
DI:Dependency Injection (依赖注入):Spring 创建对象的过程转给你,将对象依赖的属性(常量、对象、集合)通过配置设置值给该对象
IoC: 将对象的创建权,反转给了Spring容器
■ 注入:☺ 简单理解就是给对象设置值
(通过对象的setter方法【属性注入方法】、通过对象的构造器设置值【构造器注入方法】)
□ 设置值的类型:
- 常量类型(固定不变),简单类型 value
- 对象类型(引用类型) ref
- 集合类型 各自集合对应的元素<set><list>...
1、通过XML配置装配
(1)XML自动装配(不推荐)通过bean元素的属性 autowire 自动装配
✿(2)setter注入 [ 属性注入(根据类型区分)]
■(常用
) 注入常量 value
<bean id="person" class="com.shan.di_setter.Person">
<property name="name" value="shan"/>
<property name="age" value="22"/>
<property name="salary" value="10000"/>
</bean>
■(常用
) 注入对象 ref
<bean id="cat" class="com.shan.di_setter2.Cat">
<property name="name" value="kity"/>
</bean>
<bean id="person" class="com.shan.di_setter2.Person">
<property name="name" value="shan"/>
<property name="age" value="22"/>
<property name="cat" ref="cat"/>
</bean>
■ 注入集合 <set>、 <list>、<array>、 <map>、 <props>
<bean id="person" class="com.shan.di_setter3.Person">
<!-- set类型 -->
<property name="set">
<set>
<value>set1</value>
<value>set2</value>
<value>set3</value>
</set>
</property>
<!-- list类型 -->
<property name="list">
<list>
<value>list1</value>
<value>list2</value>
</list>
</property>
<!-- array类型 -->
<property name="array">
<array>
<value>array1</value>
</array>
</property>
<!-- map类型(字典类型) -->
<property name="map">
<map>
<entry key="key1" value="value1"/>
</map>
</property>
<!-- properties类型(特殊的map类型【key和value都是字符串】) -->
<property name="prop">
<value>
p1=v1
p2=v2
</value>
</property>
</bean>
● 属性的设置值是在 init 方法执行之前完成。com.shan.di_construstor
(3)构造器注入(跟属性注入差不多,常用的是属性注入
,就是将property换成constructor-arg 元素)
<!-- 构造器注入:常量类型 -->
<bean id="person" class="com.shan.di_construstor.Person">
<constructor-arg name="name" value="shan"/>
<constructor-arg name="age" value="18"/>
<constructor-arg name="salary" value="10000"/>
</bean>
<!-- 构造器注入:对象类型 -->
<bean id="cat" class="com.shan.di_construstor.Cat"/>
<bean id="person2" class="com.shan.di_construstor.Person2">
<constructor-arg name="c" ref="cat"/>
</bean>
<!-- 构造器注入:集合类型 -->
✿(4)bean元素继承 (本质是xml配置内容的拷贝)
- 通过abstract属性进行抽取
- 通过parent属性进行引入
(5) 配置数据库连接池
<!-- 配置数据库连接池 -->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/springdemo?useSSL=false"/>
<property name="username" value="root"/>
<property name="password" value="admin"/>
<property name="initialSize" value="2"/>
</bean>
✿ 动态加载配置文件(db.properties---数据库连接的配置信息)
(6)property place holder
1)要是使用标签Context,需要先引入Context的约束(在beans的基础进行修改即可):
2) context:property-placeholder 属性占位符
<!-- 从classpath的根路径 加载db.properties -->
<context:property-placeholder location="classpath:db.properties"/>
3)使用 ${} 动态引入属性值
<!-- 配置数据库连接池 -->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
<property name="initialSize" value="${jdbc.initialSize}"/>
</bean>
写在后面的话
如果你觉得一乐的文章给您带来了一些收获,可以给个三连❤️ ,一乐会一如既往地更新有价值的博客。如果文章存在错误,也欢迎大家指出。还有,如果大家有什么好的学习技巧、学习感悟,也非常欢迎大家在评论区一起交流~
最后感谢大家的支持,谢谢~