1.1Spring概述
1.1.1Spring的概述
1.1.1.1什么是Spring ?
Spring框架是由于软件开发的复杂性而创建的。Spring使用的是基本的JavaBean来完成以前只可能由EJB完成的事情。然而,Spring的用途不仅仅限于服务器端的开发。从简单性、可测试性和松耦合性角度而言,绝大部分Java应用都可以从Spring中受益。
◆目的:解决企业应用开发的复杂性
◆功能:使用基本的JavaBean代替EJB,并提供了更多的企业应用功能
◆范围:任何Java应用
Spring是一个轻量级控制反转(IoC)和面向切面(AOP)的容器框架。
- Spring:SE/EE开发的一站式框架。
- 一站式框架:有EE开发的每一层解决方案。
- WEB层 :SpringMVC
- Service层 :Spring的Bean管理,Spring声明式事务
- DAO层 :Spring的Jdbc模板,Spring的ORM模块
1.1.1.2为什么学习Spring?
Spring是一个开源框架,它由Rod Johnson创建。它是为了解决企业应用开发的复杂性而创建的。Spring使用基本的JavaBean来完成以前只可能由EJB完成的事情。然而,Spring的用途不仅限于服务器端的开发。从简单性、可测试性和松耦合的角度而言,任何Java应用都可以从Spring中受益。
Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架。
轻量——从大小与开销两方面而言Spring都是轻量的。完整的Spring框架可以在一个大小只有1MB多的JAR文件里发布。并且Spring所需的处理开销也是微不足道的。此外,Spring是非侵入式的:典型地,Spring应用中的对象不依赖于Spring的特定类。
控制反转——Spring通过一种称作控制反转(IoC)的技术促进了松耦合。当应用了IoC,一个对象依赖的其它对象会通过被动的方式传递进来,而不是这个对象自己创建或者查找依赖对象。你可以认为IoC与JNDI相反——不是对象从容器中查找依赖,而是容器在对象初始化时不等对象请求就主动将依赖传递给它。
面向切面——Spring提供了面向切面编程的丰富支持,允许通过分离应用的业务逻辑与系统级服务(例如审计(auditing)和事务(transaction)管理)进行内聚性的开发。应用对象只实现它们应该做的——完成业务逻辑——仅此而已。它们并不负责(甚至是意识)其它的系统级关注点,例如日志或事务支持。
容器——Spring包含并管理应用对象的配置和生命周期,在这个意义上它是一种容器,你可以配置你的每个bean如何被创建——基于一个可配置原型(prototype),你的bean可以创建一个单独的实例或者每次需要时都生成一个新的实例——以及它们是如何相互关联的。然而,Spring不应该被混同于传统的重量级的EJB容器,它们经常是庞大与笨重的,难以使用。
框架——Spring可以将简单的组件配置、组合成为复杂的应用。在Spring中,应用对象被声明式地组合,典型地是在一个XML文件里。Spring也提供了很多基础功能(事务管理、持久化框架集成等等),将应用逻辑的开发留给了你。
所有Spring的这些特征使你能够编写更干净、更可管理、并且更易于测试的代码。它们也为Spring中的各种模块提供了基础支持。
1.1.1.3Spring的版本
Spring3.x和Spring4.x
1.1.2Spring的入门(IOC)
- IOC: Inversion of Control(控制反转)。
- 控制反转:将对象的创建权反转给(交给)Spring。
1.1.2.1下载Spring的开发包
1.1.2.2解压Spring的开发包
- docs :Spring的开发规范和API
- libs :Spring的开发的jar和源码
- schema :Spring的配置文件的约束
1.1.2.3创建web项目,引入jar包
1.1.2.4创建接口和类
/**
* 用户管理DAO层接口
*/
public interface UserDAO {
public void save();
}
/**
* 用户管理DAO层实现类
*/
public class UserDAOImpl implements UserDAO {
private String name;
public void setName(String name) {
this.name = name;
}
public void save() {
System.out.println("UserDAOImpl执行了..."+name);
}
}
- 问题:
- 如果底层的实现切换了,需要修改源代码,能不能不修改程序源代码对程序进行扩展?
1.1.2.5将实现类交给Spring管理
约束:在spring的解压路径下spring-framework-4.2.4.RELEASE\docs\spring-framework-reference\html\xsd-configuration.html
<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的入门的配置==================== -->
<bean name="userDAO" class="come.itheima.spring.demo1.UserDAOImpl" ></bean>
</beans>
1.1.2.6编写测试类
/**
* Spring的入门
*/
public class SpringDemo1 {
@Test
/**
* 传统方式的调用
*/
public void demo1(){
UserDAOImpl userDAO = new UserDAOImpl();
userDAO.setName("王东");
userDAO.save();
}
@Test
/**
* Spring的方式的调用
*/
public void demo2(){
// 创建Spring的工厂
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDAO userDAO = (UserDAO) applicationContext.getBean("userDAO");
userDAO.save();
}
@Test
/**
* 加载磁盘上的配置文件
*/
public void demo3(){
ApplicationContext applicationContext = new FileSystemXmlApplicationContext("C:\\applicationContext.xml");
UserDAO userDAO = (UserDAO) applicationContext.getBean("userDAO");
userDAO.save();
}
}
1.1.2.7IOC和DI(*****)
- IOC:控制反转,将对象的创建权反转给了Spring。
- DI:依赖注入,前提必须有IOC的环境,Spring管理这个类的时候将类的依赖的属性注入(设置)进来。
- 面向对象的时候
- 依赖:B类依赖A类
Class A{
}
Class B{
public void xxx(A a){
}
}
- 继承:is a
Class A{
}
Class B extends A{
}
- 聚合:has a
1.2Spring的工厂类
1.2.1Spring的工厂类
1.2.1.1Spring工厂类的结构图
- ApplicationContext继承BeanFactory。
1.2.1.2BeanFactory :老版本的工厂类
- BeanFactory:调用getBean的时候,才会生成类的实例。
1.2.1.3ApplicationContext :新版本的工厂类
- ApplicationContext:加载配置文件的时候,就会将Spring管理的类都实例化。
- ApplicationContext有两个实现类
- ClassPathXmlApplicationContext :加载类路径下的配置文件
- FileSystemXmlApplicationContext :加载文件系统下的配置文件
1.3Spring的配置
1.3.1XML的提示配置
1.3.1.1Schema的配置
1.3.2Bean的相关的配置
1.3.2.1<bean>标签的id和name的配置
- id :使用了约束中的唯一约束。里面不能出现特殊字符的。
- name :没有使用约束中的唯一约束(理论上可以出现重复的,但是实际开发不能出现的)。里面可以出现特殊字符。
- Spring和Struts1框架整合的时候
- <bean name=”/user” class=””/>
1.3.2.2Bean的生命周期的配置(了解)
- init-method :Bean被初始化的时候执行的方法
- destroy-method :Bean被销毁的时候执行的方法(Bean是单例创建,工厂关闭)
<!-- Spring的sBean的生命周期的配置=========== -->
<bean id="customerDAO" class="come.itheima.spring.demo2.CustomerDAOImpl" init-method="setup" destroy-method="destroy"/>
1.3.2.3Bean的作用范围的配置(重点)
- scope :Bean的作用范围
- singleton :默认的,Spring会采用单例模式创建这个对象。
- prototype :多例模式。(Struts2和Spring整合一定会用到)
- request :应用在web项目中,Spring创建这个类以后,将这个类存入到request范围中。
- session :应用在web项目中,Spring创建这个类以后,将这个类存入到session范围中。
- globalsession :应用在web项目中,必须在porlet环境下使用。但是如果没有这种环境,相对于session。
<!-- Spring的sBean的生命周期的配置=========== -->
<bean id="customerDAO" class="come.itheima.spring.demo2.CustomerDAOImpl" scope="prototype" />
1.4Spring的Bean管理(XML方式)
1.4.1Spring的Bean的实例化方式(了解)
Bean已经都交给Spring管理,Spring创建这些类的时候,有几种方式:
1.4.1.1无参构造方法的方式(默认)
- 编写类
/**
* 无参数构造方法方式
*/
public class Bean1 {
public Bean1() {
super();
System.out.println("Bean1的无参数的构造方法执行了...");
}
}
- 编写配置
<!-- 无参数构造方法 -->
<bean id="bean1" class="com.itheima.spring.demo3.Bean1"></bean>
1.4.1.2静态工厂实例化的方式
- 编写类
/**
* 静态工厂实例化方式
*/
public class Bean2 {
}
- 编写Bean2的静态工厂
/**
* Bean2的静态工厂
*/
public class Bean2Factory {
public static Bean2 createBean2(){
System.out.println("Bean2Factory中方法执行了...");
return new Bean2();
}
}
- 配置
<!-- 静态工厂实例化 -->
<bean id="bean2" class="com.itheima.spring.demo3.Bean2Factory" factory-method="createBean2"/>
1.4.1.3实例工厂实例化的方式
- 编写类
/**
* 实例工厂实例化的方式
*/
public class Bean3 {
}
- Bean3的实例工厂
/**
* Bean3的实例工厂
*/
public class Bean3Factory {
public Bean3 createBean3(){
System.out.println("Bean3的实例工厂执行了...");
return new Bean3();
}
}
- 配置
<!-- 实例工厂实例化 -->
<bean id="bean3Factory" class="com.itheima.spring.demo3.Bean3Factory"></bean>
<bean id="bean3" factory-bean="bean3Factory" factory-method="createBean3"></bean>
1.6.2Spring的属性注入
1.6.2.1构造方法的方式的属性注入
- 构造方法的属性注入
<!-- 构造方法的方式 -->
<bean id="car" class="com.itheima.spring.demo4.Car">
<constructor-arg name="name" value="宝马"/>
<constructor-arg name="price" value="800000"/>
</bean>
public class Car {
private String name;
private Double price;
public Car(String name, Double price) {
super();
this.name = name;
this.price = price;
}
@Override
public String toString() {
return "Car [name=" + name + ", price=" + price + "]";
}
}
demo:
@Test
/**
* 构造方法方式的属性注入
*/
public void demo1(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
Car car = (Car) applicationContext.getBean("car");
System.out.println(car);
}
1.4.2.2Set方法的方式的属性注入
- Set方法的属性注入
<!-- set方法的方式 -->
<bean id="car2" class="com.itheima.spring.demo4.Car2">
<property name="name" value="奔驰"/>
<property name="price" value="1000000"/>
</bean>
- Set方法设置对象类型的属性
<!-- set方法注入对象类型的属性 -->
<bean id="employee" class="com.itheima.spring.demo4.Employee">
<!-- value:设置普通类型的值,ref:设置其他的类的id或name -->
<property name="name" value="涛哥"/>
<property name="car2" ref="car2"/>
</bean>
public class Car2 {
private String name;
private Double price;
public void setName(String name) {
this.name = name;
}
public void setPrice(Double price) {
this.price = price;
}
@Override
public String toString() {
return "Car2 [name=" + name + ", price=" + price + "]";
}
}
public class Employee {
private String name;
private Car2 car2;
public void setName(String name) {
this.name = name;
}
public void setCar2(Car2 car2) {
this.car2 = car2;
}
@Override
public String toString() {
return "Employee [name=" + name + ", car2=" + car2 + "]";
}
}
demo:
@Test
/**
* set方法方式的属性注入
*/
public void demo2(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
Car2 car2 = (Car2) applicationContext.getBean("car2");
System.out.println(car2);
}
@Test
/**
* set方法注入对象类型
*/
public void demo3(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
Employee employee = (Employee) applicationContext.getBean("employee");
System.out.println(employee);
}
1.4.2.3P名称空间的属性注入(Spring2.5以后)
- 通过引入p名称空间完成属性的注入:
- 写法:
- 普通属性 p:属性名=”值”
- 对象属性 p:属性名-ref=”值”
- P名称空间的引入
- 使用p名称空间
<!-- 改为p名称空间的方式 -->
<bean id="car2" class="com.itheima.spring.demo4.Car2" p:name="奇瑞QQ" p:price="30000"></bean>
<bean id="employee" class="com.itheima.spring.demo4.Employee" p:name="王东" p:car2-ref="car2"></bean>
1.4.2.4SpEL的属性注入(Spring3.0以后)
- SpEL:Spring Expression Language,Spring的表达式语言。
- 语法:
- #{SpEL}
public class Employee {
private String name;
private Car2 car2;
public void setName(String name) {
this.name = name;
}
public void setCar2(Car2 car2) {
this.car2 = car2;
}
@Override
public String toString() {
return "Employee [name=" + name + ", car2=" + car2 + "]";
}
}
HTML:
<!-- SpEL的属性注入 -->
<bean id="carInfo" class="com.itheima.spring.demo4.CarInfo">
</bean>
<bean id="car2" class="com.itheima.spring.demo4.Car2">
<property name="name" value="#{carInfo.name}"></property>
<property name="price" value="#{carInfo.calculatorPrice()}"></property>
</bean>
<bean id="employee" class="com.itheima.spring.demo4.Employee">
<property name="name" value="#{'赵洪'}"></property>
<property name="car2" value="#{car2}"></property>
</bean>
1.6.3集合类型属性注入(了解)
1.6.3.1配置
<!-- Spring的集合属性的注入============================ -->
<!-- 注入数组类型 -->
<bean id="collectionBean" class="com.itheima.spring.demo5.CollectionBean">
<!-- 数组类型 -->
<property name="arrs">
<list>
<value>王东</value>
<value>赵洪</value>
<value>李冠希</value>
</list>
</property>
<!-- 注入list集合 -->
<property name="list">
<list>
<value>李兵</value>
<value>赵如何</value>
<value>邓凤</value>
</list>
</property>
<!-- 注入set集合 -->
<property name="set">
<set>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</set>
</property>
<!-- 注入Map集合 -->
<property name="map">
<map>
<entry key="aaa" value="111"/>
<entry key="bbb" value="222"/>
<entry key="ccc" value="333"/>
</map>
</property>
</bean>
1.5Spring的分模块开发的配置
1.5.1分模块配置
1.5.1.1在加载配置文件的时候,加载多个
1.5.1.2在一个配置文件中引入多个配置文件
<import resource="applicationContext2.xml"/>
1.8CRM的综合案例(Spring整合Web项目)
1.8.1代码实现
1.8.1.1创建数据库和表
1.8.1.2创建web项目,引入jar包
- 引入struts2的开发的jar包
- 引入struts2的配置文件
- web.xml
- struts.xml
1.8.1.3引入页面
1.8.1.4编写Action类
1.8.1.5配置Action
1.8.1.6修改页面提交到Action
1.8.1.7编写Action的save方法
- 不在Action中直接创建Service,将Service交给Spring管理。
- 创建Service接口和实现类
1.8.1.8引入Spring的环境
- 引入jar包
- 引入配置文件
1.8.1.9将Service交给Spring
1.8.1.10在Action中调用Service
1.8.1.11编写DAO并且完成配置
- 编写DAO的接口和实现类
- 将DAO交给Spring管理
1.8.1.12在Service中使用DAO
1.8.2问题描述
1.8.2.1程序问题
- 每次请求都会创建一个Spring的工厂,这样浪费服务器资源,应该一个项目只有一个Spring的工厂。
- 在服务器启动的时候,创建一个Spring的工厂。
- 创建完工厂,将这个工厂类保存到ServletContext中。
- 每次使用的时候都从ServletContext中获取。
- *****使用ServletContextListener
- 监听ServletContext对象的创建和销毁。
1.8.2.2解决方案:使用Spring核心监听器ContextLoaderListener(整合web项目)
- 引入jar包
spring-web-4.2.4.RELEASE.jar
- 在web.xml配置监听器
- 在Action中获取工厂