Spring引入
* spring 是什么?
Spring是分层的 Java SE/EE应用 full-stack 轻量级开源框架,以 IoC(Inverse Of Control:
反转控制)和 AOP(Aspect Oriented Programming:面向切面编程)为内核,提供了展现层 Spring
MVC 和持久层 Spring JDBC 以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多
著名的第三方框架和类库,逐渐成为使用最多的Java EE 企业应用开源框架。
* 程序的耦合
耦合:程序之间的依赖关系
包括了类与类之间的依赖,方法之间的依赖
解耦:降低程序之间的依赖关系,提升程序的灵活性与可扩展性
实际开发中:应该做到编译期不依赖,运行时才会依赖
解耦的思路:
第一步:使用反射来创建对象,从而避免使用new关键字
第二步:通过读取配置文件来获取要创建的对象全限定类名
* 三层架构代码中存在的问题
业务层调用持久层的时候的new dao的接口
在表现层调用业务层的时候new service的接口
解决办法: 使用工厂模式解耦(一个创建bean对象的工厂)
Bean
1. Bean:在计算机英语中,有可重用组件的含义
2. JavaBean:用java语言编写的可重用组件
JavaBean > 实体类
3. POJO(Plain Ordinary Java Object)即普通Java类
4. 可以认为POJO就一个不带方法而单纯承载数据实体Bean;EJB是一个包含了POJO的超集、
包含POJO之外的,具有方法、实现、功能这一群组件。
方法:
1. 一个配置文件配置service与dao的内容(xml/properties)
2. 通过读取配置文件来获取要创建的对象全限定类名,反射加载
3.优化:使用单例
Spring的IOC(Inversion Of Control)
* IOC:控制反转
简单理解:让spring来控制对象
* 入门案例:
1. 导入依赖
2. 创建配置文件,导入约束,并写入相关的bean标签(配置id与class)
3. 从容器中获取容器对象
3.1 先获取容器
ApplicationContext ac = new ClassPathXmlApplicationContext("配置文件")
3.2 从容器中根据id或者name获取bean对象
User user = ac.getBean("user", User.class);
System.out.println(user);//User{userName='null', age=null, birthday=null}
* ApplicationContext的三个常用的实现类
ClassPathXmlApplicationContext:它可以加载类路径下的配置文的件,要求配置文件必须在类路径下,不在就加载不了
FileSystemXmlApplicationContext:它可以加载磁盘任意路径下的配置文件 (必须有访问权限)
AnnotationConfigApplicationContext:它是用于读取注解的配置
* 核心容器的两个接口引发出的问题
ApplicationContext:(单例对象试用),多使用此接口
它在构建核心容器的时候,创建对象采取的是立即加载的方式,也就是说,只要一读取完配置文件就马上
就创建配置文件中的配置的对象
BeanFactory:(多例对象适用)
它在构建核心容器的时候,采取的策略是延迟加载的方式,也就是说,什么时候根据id获取对象了,什么时候才会
真正的创建对象
* Spring对bean的管理细节
1. 创建bean的三种方式:
1.1 第一种方式:使用默认构造函数创建
在spring的配置文件中使用bean标签,配置id与class属性之后,且没有其他属性和标签时
采用的就是默认构造函数创建bean对象,测试如果类中没有默认构造函数,则对象无法创建
<bean id="service" class="com.qin.service.impl.IUserServiceImpl"></bean>
1.2 第二种方式:使用普通工厂中的方法来创建对象(使用某个类中的方法创建对象,并存入spring容器)
<bean id="instanceFactory" class="com.qin.factory.instanceFactory"></bean>(要先有对象才能调用方法)
<bean id="serviceFactory" class="com.qin.factory.instanceFactory" factory-method="getAccount">
</bean>
1.3 第三种方式:使用工厂中的静态方法创建对象(使用某个类中的静态方法创建对象,并存入spring容器)
<bean id="serviceFactory" class="com.qin.factory.StaticFactory" factory-method="getAccount">
</bean>
静态方法是不需要对象的,是可以通过全类名调用的,所以不需要对象
2. bean的作用范围
bean标签的scope属性
作用:用于指定bean 的作用范围
取值:
singleton:单例的(默认值)
prototype:多例的
request:作用于wen应用的请求范围
session:作用于wen应用的会话范围
global-session:作用于集群环境的会话范围(全局会话范围),当不是集群的时候,相当于session
3. bean对象的生命周期
单例对象:
出生:当容器创建时候对象出生
存在:只要容器在,对象就一直活着
死亡:容器销毁,对象消亡
总结:单例对象的生命周期和容器相同
多例对象:
出生:当使用对象时,spring框架为我们创建
存在:对象还在使用的过程中就一直存活
死亡:当对象长时间不用,且没有别的对象引用时,由java的垃圾回收器回收
Spring依赖注入:DI(依赖关系的维护)
* 能注入的数据:三类
1. 基本类型和string :
2. 其他bean类型(在配置文件中或者注解配置过的bean)
3. 复杂类型/集合类型
* 注入的方式:有三种
1. 使用构造函数提供(不常用)
使用的标签:constructor-arg
标签出现的位置:bean标签的内部
标签的属性:
type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
index:用于指定要注入的数据给构造函数中指定索引位置的参数赋值,索引的位置是从0开始的
name:用于指定给构造函数中指定名称的参数赋值(常用)
=============以上三个用于指定给构造函数中的哪个参数赋值==================
value:用于提供基本类型和string类型的数据
ref:用于指定其他的bean类型数据.他指的就是在spring的核心容器中出现过的bean对象
优势:
在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功.
弊端:
改变了bean对象的实例化方式,使我们在创建对象时,如果不使用这些数据也必须提供.
2. 使用set方法提供(常用)
* 首先必须有set方法
涉及的标签:property
出现的位置:bean标签的内部
标签的属性:
name:用于指定注入时调用的set方法名称
value:用于提供基本类型和string类型的数据
ref:用于指定其他的bean类型数据.他指的就是在spring的核心容器中出现过的bean对象
优势:创建对象时没有明确的限制,可以直接使用默认构造函数
弊端:如果某个成员必须有值,则获取对象是有可能set方法没有执行
3. 使用注解提供
* 示例:
1. 构造函数注入
<bean id="userDao" class="com.qin.dao.impl.IUserDaoImpl">
<constructor-arg name="userName" value="sakura"></constructor-arg>
<constructor-arg name="age" value="18"></constructor-arg>
<constructor-arg name="birthday" ref="date"></constructor-arg>
</bean>
<bean id="date" class="java.util.Date"></bean>
2. 使用set方法注入
<bean id="userDao1" class="com.qin.dao.impl.IUserDaoImpl1">
<property name="age" value="19"></property>
<property name="userName" value="tom"></property>
<property name="birthday" ref="date"></property>
</bean>
<bean id="date" class="java.util.Date"></bean>
3. 对集合属性进行set方式注入
<bean id="userDao2" class="com.qin.dao.impl.IUserDaoImpl2">
<property name="myStrs">
<array>
<value>string</value>
<value>string</value>
<value>string</value>
</array>
</property>
<property name="myList">
<list>
<value>list</value>
<value>list</value>
<value>list</value>
</list>
</property>
<property name="mySet">
<set>
<value>set</value>
<value>set</value>
<value>set</value>
</set>
</property>
<property name="myMap">
<map>
<entry key="map1">
<value>map1</value>
</entry>
<entry key="map2" value="map2"></entry>
<entry key="map3" value="map3"></entry>
</map>
</property>
<property name="myProps">
<props>
<prop key="prop">123</prop>
<prop key="prop">123</prop>
<prop key="prop">123</prop>
</props>
</property>
</bean>
Spring的基于注解开发
* IOC在XML的配置:<bean id="accountService" class="com.qin.service.impl.AccountService" scope=""
init-method="" destory-method="">
<property name="" value="" |ref=""></property>
</bean>
* 注解分类:
1. 用于创建对象的
它们的作用就和xml配置中编写一个<bean>标签实现的功能是一样的
@Component:
作用:
用于把当前类对象存入Spring容器中
属性:
value:用于指定bean中的id.当我们不写时,他的默认值是当前类名的,且首字母小写
@Controller:一般用在表现层
@Service:一般用在业务层
@Repository:一般用在持久层
以上三个注解的作用和属性与Component是一模一样的
他们三个是spring为我们提供明确的三层使用的注解,使我们的三层对象更加清晰
在xml文件的配置:
<context:component-scan base-package="com.qin"></context:component-scan>
作用:告知spring在创建容器时要扫描的包,配置所需要的标签不是beans约束中,而是在context
的名称空间和约束中
约束配置:
<?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
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
2. 用于注入数据的
它们的作用就和在xml配置文件中的bean标签中写一个<property>标签的作用是一样的
@Autowired
作用:(自动按照类型注入)
如果ioc容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功
如果ioc容器中没有任何bean类型和要注入的变量类型一致,将会注入失败
如果ioc容器中有多个bean类型和要注入的变量类型一致,会先根据类型寻找,再根据变量名找
出现位置:
变量上,方法上(在set方法上也可以使用);
细节:
在使用注解注入时,set方法就不是必须的
@Qualifier
* 配合@Autowired使用
* 也可以存在于参数上,配合指定构造中要传递的bean对象
作用:
在按照类中注入的基础之上再按照名称注入.他在给类成员注入时不能单独使用,但是在给方法参数注入时可以
属性:
value:用于指定注入bean的id
@Resource
* jdk1.9不能使用
作用:
直接按照bean的id注入,它可以单独使用
属性:
name:用于指定bean 的id
* 以上三个注入都只能注入其他bean类型,而基本类型和string类型无法使用上述注解实现,集合类型也可以用Autowired实现
@Value
作用:
用于注入基本类型和string类型的数据
属性:
value:用于指定数据的值.他可以使用spring和spEL(也就是spring的el表达式)
spEL的写法:#{表达式}
#{}:去容器中找id为花括号内容对应的值
${}:去配置文件中读取花括号对应的值
3. 用于改变作用范围的
它们的作用就和在xml配置文件中的bean标签中使用scope属性实现的功能是一样的
@Scope
作用:
用于指定bean的作用范围
属性:
value:指定范围的取值.常用取值:singleton prototype
4. 和生命周期相关
它们的作用就和在xml配置文件中的bean标签中使用init-method和destory-method的作用是一样的
@PreDestroy
作用:
用于指定销毁方法
@PostConstruct
作用:
用于指定初始化方法
* spring的新注解(可以使用纯注解的形式开发)
* 创建一个类.替换核心配置文件
@Configuration
作用:
指定当前类是一个配置类
细节:
当配置类作为AnnotationConfigApplicationContext对象创建的参数时,可以不写此注解
但不是任何时候都可以不写(把注解文件分开写,另一个没有读的配置文件必须写)
@ConponentScan
作用:
用于通过注解指定spring在创建容器时要扫面的包
属性:
value:它和basePackage的作用是一样的,都是用于指定创建容器时要扫面的包
我们使用次注解就等同于在xml中配置了:
<context:component-scan base-package="com.qin"></context:component-scan>
@Bean
作用:
用于把当前方法的的返回值作为bean对象存入spring的ioc容器中
属性:
name:用于指定bean的id,当不写时,默认值是当前方法的名称
细节:
当我们使用注解来配置方法时,如果方法有参数,spring框架会去容器中查找有没有可用的bean对象,查找的方式和
@Autowired注解的作用一样的
@Import
作用:
用于导入其他的配置类
属性:
value:用于指定其他配置类的字节码文件
当我们使用import的注解之后,有import注解的类就是父配置类,而导入的都是子配置类
@PropertySource
作用:
用于指定properties文件的位置
属性:
value:指定文件的名称以及文件的路径
关键字:classpath,表示类路径下
例子:
@PropertySource("classpath:JdbcConfig.properties")
*没有xml文件时测试就要修改ApplicationContext的创建方式
ApplicationContext ac = new AnnotationConfigApplicationContext("被@Configuration注解过的类".class)
抽取配置数据库文件
1.抽取文件:
jdbcConfig.properties
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/spring
jdbc.username=root
jdbc.password=root
2. 在Spring的配置文件中配置:
<context:property-placeholder location="jdbcConfig.properties"></context:property-placeholder>
使用注解的配置为:
@PropertySource("classpath:jdbcConfig.properties")
3. 在原来的配置文件中使用Spring的el表达式
${}来表示
<property name="driverClass" value="${jdbc.driver}"></property>
<property name="jdbcUrl" value="${jdbc.url}"></property>
<property name="user" value="${jdbc.username}"></property>
<property name="password" value="${jdbc.password"></property>