以前也学习过Spring框架,不过好久没用,当时学完也没做什么总结,都忘的差不多了,今天再从头开始学习一遍。无论是SSH还是SSM都离不开Spring,所以Spring还是很重要的,对于一个想要从事JavaEE开发的人,一定要好好学习Spring框架。Spring的学习计划如下:
l 第一天:Spring的概述、SpringIOC入门(XML)、Spring的Bean管理、Spring属性注入
l 第二天:Spring的IOC的注解方式、Spring的AOP开发(XML)
l 第三天:Spring的AOP的注解开发、Spring的声明式事务、JdbcTemplate。
l 第四天:SSH的整合、HibernateTemplate的使用、OpenSessionInViewFilter的使用。
Spring是一个开放源代码的设计层面框架,他解决的是业务逻辑层和其他各层的松耦合问题,因此它将面向接口的编程思想贯穿整个系统应用。Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson创建。简单来说,Spring是一个分层的JavaSE/EE full-stack(一站式) 轻量级开源框架。
l Spring:JavaSE/EE开发的一站式框架。
n 一站式框架:JavaSE/EE开发的每一层的解决方案。
u WEB层 :SpringMVC
u Service层 :Spring的Bean管理,Spring声明式事务
u DAO层 :Spring的Jdbc模板,Spring的ORM模块
轻量——从大小与开销两方面而言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也提供了很多基础功能(事务管理、持久化框架集成等等),将应用逻辑的开发留给了你。
MVC——Spring的作用是整合,但不仅仅限于整合,Spring 框架可以被看做是一个企业解决方案级别的框架。客户端发送请求,服务器控制器(由DispatcherServlet实现的)完成请求的转发,控制器调用一个用于映射的类HandlerMapping,该类用于将请求映射到对应的处理器来处理请求。HandlerMapping 将请求映射到对应的处理器Controller(相当于Action)在Spring 当中如果写一些处理器组件,一般实现Controller 接口,在Controller 中就可以调用一些Service 或DAO 来进行数据操作 ModelAndView 用于存放从DAO 中取出的数据,还可以存放响应视图的一些数据。如果想将处理结果返回给用户,那么在Spring 框架中还提供一个视图组件ViewResolver,该组件根据Controller 返回的标示,找到对应的视图,将响应response 返回给用户。
所有Spring的这些特征使你能够编写更干净、更可管理、并且更易于测试的代码。它们也为Spring中的各种模块提供了基础支持。
Spring3.x和Spring4.x,今天学习Spring4.x
l IOC: Inversion of Control(控制反转)。
n 控制反转:将对象的创建权反转给(交给)Spring。
官网:http://spring.io/
l docs :Spring的开发规范和API
l libs :Spring的开发的jar和源码
l schema :Spring的配置文件的约束
packagecom.albertyy.spring.demo1;
/**
* 用户管理DAO层接口
* @author yxy
*
*/
publicinterfaceUserDAO {
publicvoidsave();
}
packagecom.albertyy.spring.demo1;
/**
* 用户管理DAO层实现类
* @author yxy
*
*/
publicclassUserDAOImpl implements UserDAO {
private String name;
publicvoidsetName(String name) {
this.name = name;
}
@Override
publicvoid save() {
System.out.println("UserDAOImpl执行了..."+name);
}
}
l 问题:
n 如果底层的实现切换了,需要修改源代码,能不能不修改程序源代码对程序进行扩展?
在spring的解压路径下:spring-framework-4.2.4.RELEASE\docs\spring-framework-reference\html\xsd-configuration.html
在项目src路径下面新建applicationContext.xml配置文件:
<?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的入门的配置==================== -->
<bean name="userDAO" class="com.albertyy.spring.demo1.UserDAOImpl">
<property name="name" value="李慢慢" />
</bean>
</beans>
@Test
/**
* 传统方式的调用
*/
publicvoid demo1(){
UserDAOImpl userDAO = new UserDAOImpl();
userDAO.setName("王婉瑜");
userDAO.save();
}
@Test
/**
* Spring的方式的调用
*/
publicvoid demo2(){
// 创建Spring的工厂
ApplicationContextapplicationContext = newClassPathXmlApplicationContext("applicationContext.xml");
UserDAO userDAO = (UserDAO) applicationContext.getBean("userDAO");
userDAO.save();
}
l IOC:控制反转,将对象的创建权反转给了Spring。
n DI:依赖注入,前提必须有IOC的环境,Spring管理这个类的时候将类依赖的属性注入(设置)进来。
l 面向对象的时候类和类的关系
n 依赖
Class A{
}
Class B{
public void xxx(A a){
}
}
n 继承:(is a)
Class A{
}
Class B extendsA{
}
n 聚合:(has a)
l ApplicationContext继承BeanFactory。
l BeanFactory:调用getBean的时候,才会生成类的实例。
l ApplicationContext:加载配置文件的时候,就会将Spring管理的类都实例化。
l ApplicationContext有两个实现类
n ClassPathXmlApplicationContext :加载类路径下的配置文件
n FileSystemXmlApplicationContext :加载文件系统下的配置文件
l id :使用了约束中的唯一约束,里面不能出现特殊字符。
l name :没有使用约束中的唯一约束(理论上可以出现重复的,但是实际开发中是不能出现重复的)。里面可以出现特殊字符。
n Spring和Struts1框架整合的时候
n <bean name=”/user” class=””/>
l init-method :Bean被初始化的时候执行的方法
l destroy-method :Bean被销毁的时候执行的方法(Bean是单例创建,工厂关闭)
<!-- Spring的sBean的生命周期的配置=========== -->
<bean id="customerDAO" class="com.albertyy.spring.demo2.CustomerDAOImpl"scope="prototype" init-method="setup"destroy-method="destroy"/>
l scope:Bean的作用范围
n singleton :默认的,Spring会采用单例模式创建这个对象。
n prototype :多例模式。(Struts2和Spring整合一定会用到)
n request :应用在web项目中,Spring创建这个类以后,将这个类存入到request范围中。
n session :应用在web项目中,Spring创建这个类以后,将这个类存入到session范围中。
n globalsession :应用在web项目中,必须在porlet环境下使用。但是如果没有这种环境,相对于session。
Bean已经都交给Spring管理,Spring创建这些类的时候,有几种方式:
l 编写类
packagecom.albertyy.spring.demo3;
/**
* 无参数构造方法方式
* @author yxy
*
*/
publicclass Bean1 {
public Bean1() {
super();
System.out.println("Bean1的无参数的构造方法执行了...");
}
}
l 编写配置
<!-- 无参构造方法-->
<bean id="bean1" class="com.albertyy.spring.demo3.Bean1"></bean>
l 编写Bean2的静态工厂
packagecom.albertyy.spring.demo3;
/**
*Bean2的静态工厂
* @author yxy
*
*/
publicclassBean2Factory {
publicstatic Bean2createBean2(){
System.out.println("Bean2Factory中方法执行了...");
returnnew Bean2();
}
}
l 配置
<!-- 静态工厂实例化 -->
<bean id="bean2" class="com.albertyy.spring.demo3.Bean2Factory"factory-method="createBean2"></bean>
l Bean3的实例工厂
packagecom.albertyy.spring.demo3;
/**
*Bean3的实例工厂
* @author yxy
*
*/
publicclassBean3Factory {
public Bean3createBean3(){
System.out.println("Bean3的实例工厂执行了...");
returnnew Bean3();
}
}
l 配置
<!-- 实例工厂实例化 -->
<bean id="bean3Factory" class="com.albertyy.spring.demo3.Bean3Factory"></bean>
<bean id="bean3" factory-bean="bean3Factory" factory-method="createBean3"></bean>
l 构造方法的属性注入
<!-- 构造方法的方式 -->
<bean id="car" class="com.albertyy.spring.demo4.Car">
<constructor-arg name="name" value="宝马"/>
<constructor-arg name="price" value="800000"/>
</bean>
l Set方法的属性注入
<!-- set方法的方式 -->
<beanid="car2" class=" com.albertyy.spring.demo4.Car2">
<propertyname="name" value="奔驰"/>
<propertyname="price" value="1000000"/>
</bean>
l Set方法设置对象类型的属性
<!-- set方法注入对象类型的属性 -->
<beanid="employee" class=" com.albertyy.spring.demo4.Employee">
value:设置普通类型的值,ref:设置其他的类的id或name
<propertyname="name" value="涛哥"/>
<propertyname="car2" ref="car2"/>
</bean>
l 通过引入p名称空间完成属性的注入:
n 写法:
u 普通属性 p:属性名=”值”
u 对象属性 p:属性名-ref=”值”
l P名称空间的引入
l 使用p名称空间
<!-- 改为p名称空间的方式 -->
<beanid="car2" class=" com.albertyy.spring.demo4.Car2"p:name="奇瑞QQ" p:price="30000"></bean>
<beanid="employee" class=" com.albertyy.spring.demo4.Employee"p:name="王东" p:car2-ref="car2"></bean>
l SpEL:Spring ExpressionLanguage,Spring的表达式语言。
n 语法:
u #{SpEL}
<!-- SpEL的属性注入 -->
<bean id="carInfo" class="com.albertyy.spring.demo4.CarInfo">
</bean>
<bean id="car2" class="com.albertyy.spring.demo4.Car2">
<property name="name" value="#{carInfo.name}"></property>
<property name="price" value="#{carInfo.calculatorPrice()}"></property>
</bean>
<bean id="employee" class="com.albertyy.spring.demo4.Employee">
<property name="name" value="#{'赵洪'}"></property>
<property name="car2" value="#{car2}"></property>
</bean>
<!--Spring的集合属性的注入============================-->
<!--注入数组类型-->
<beanid="collectionBean" class="com.albertyy.spring.demo5.CollectionBean">
<!--数组类型-->
<propertyname="arrs">
<list>
<value>王东</value>
<value>赵洪</value>
<value>李冠希</value>
</list>
</property>
<!--注入list集合 -->
<propertyname="list">
<list>
<value>李兵</value>
<value>赵如何</value>
<value>邓凤</value>
</list>
</property>
<!--注入set集合 -->
<propertyname="set">
<set>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</set>
</property>
<!--注入Map集合 -->
<propertyname="map">
<map>
<entrykey="aaa" value="111"/>
<entrykey="bbb" value="222"/>
<entrykey="ccc" value="333"/>
</map>
</property>
</bean>
ApplicationContextapplicationContext = newClassPathXmlApplicationContext("applicationContext.xml","applicationContext2.xml");
<importresource="applicationContext2.xml"/>
源码:链接: https://pan.baidu.com/s/1RKJ5Cd1XokL-rRpgIdFjkg提取码: 7vv2