标签:应用程序开发 属性 通过 etag 套件 enc 加载 上下文 getname
Spring框架复习之一 <1>Spring是一个分层的javaSE/EE full-stack(一站式) 轻量级开源框架。
<2>Spring的核心是控制反转(IOC)和面向切面(AOP),
<3>IOC和DI:
* IOC(Inversion Of Control)控制反转:将对象的创建权交给spring,提高解耦合性
* DI(Dependency Injection 依赖注入):
在IOC的环境下,spring创建这个类的过程中,spring将类的依赖属性也设置进去。
<4>Spring的IOC的实现原理
* 传统创建dao对象方式
&原始方式:UserDaoImpl ud=new UserDaoImpl();
& 面向接口方式:
UserDao ud=new UserDaoImpl();
此方式 缺点为:接口和实现类有耦合,两者联系过繁,来回切换底层实现类。
好的程序设计应满足OCP原则 Open-Close Principle,
对程序扩展是Open的,对修改源码是close的。
* Spring的工厂模式:
Spring的工厂模式(工厂对象+反射+配置文件)控制对象的创建,从而完成接口和实现类的耦合
UserDao ud=new UserDaoHibernateImpl();
此方式,接口和实现类之间没有耦合,sprig工厂和接口有耦合。
分层结构允许使用者选择哪一个组件,同时为J2EE应用程序开发提供集成的框架。
JavaEE开发分成三层架构:
* WEB层:springMVC
* 业务层:Bean管理(IOC)
* 持久层:Spring的JDBC模版,ORM模版
<1>方便解耦,简化开发
Spring就是一个大工厂,维护管理所有对象的创建和关系依赖.
<2>支持AOP编程(纵向重复代码横向抽取,Servlet的过滤器filter 和 Action的拦截器interceptor)
提供AOP面向切面编程,方便实现对程序的权限拦截,运行监控等功能
<3>支持声明式事务
只需通过配置就可以完成对事物的管理,而无需手动进行。
<4>方便程序测试
Spring支持Junit4,可通过注解方便测试Spring程序
<5>方便集成各种优秀框架
Spring不排斥各种优秀的开源框架,如对Struts2,Hibernate,Mybatis,Quartz等直接支持。
<6>降低JavaEE API的使用难度
Spring对JavaEE的一些难用API(JDBC,JavaMail,远程调用等),都提供封装,降低这些API的应用难度。
* docs:API规范帮助文档
* libs:jar包和源码
六个基本包
& 四个核心包:
<1>spring-core-4.2.4.RELEASE.jar --core
<2>spring-context-4.2.4.RELEASE.jar --context
<3>spring-beans-4.2.4.RELEASE.jar --beans
<4>spring-expression-4.2.4.RELEASE.jar --expression
& 两个基本包:
<1>com.springsource.org.apache.commons.logging-1.1.1.jar --logging
<2>com.springsource.org.apache.log4j-1.2.15.jar --log4j
* schema:存放约束文件
<1>引入约束
* 约束声明获取路径:
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">
</beans>
值得一提的是此约束导入也可以在eclipse中进行配置实现。
<2>设置STS(Spring Tools Suite Spring框架工具套件) 的XML提示,Schema配置。
获取spring-beans-4.2.xsd文件路径:
spring-framework-4.2.4.RELEASE\schema\beans\spring-beans-4.2.xsd
<3>什么是xsd文件?
* XSD ( XML Schemas Definition )XML结构定义, 是DTD的替代品。
* XSD是替代DTD的原因:
&比DTD可扩展性强,丰富和有用。
& 用XML书写,支持数据类型,支持命名空间。
* XML Schema(XSD语言):
Schema本身是一个XML文档,它符合XML语法结构,可以用通用的XML解析器解析它。
Schema指定一个XML文档所允许的结构和内容,并可据此检查一个XML文档是否是有效的。
<3>基本标签<bean>介绍:
* Bean元素描述需要 被spring容器管理的对象
其属性特性为:
class属性:被管理对象的完整类名
name属性:给被管理的对象起一个名字,获得对象时根据该名称获得对象,
可以重复,命名中可包含特殊字符
id属性:与name属性一模一样(是早期产生的),但名称不可重复,
不能使用特殊字符,尽量使用name属性
* 模板示例:
<beans ..................................>
<bean id="" name="" class="">
</bean>
</beans>
<4>简单测试
代码示例:
public interface TestDao {public void sayHello();}
public class TestDaoImpl implements TestDao{
@Override
public void sayHello() {
System.out.println("hello spring!");
}
}
配置示例:
<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">
<bean id="testDao" class="flower.test.spring.dao.TestDaoImpl"></bean>
</beans>
<5>Spring的IOC控制反转,管理对象的方式。
方式一 空参构造创建:
配置文件:<bean name="user" class="cn.ita.bean.User"></bean>
相应的测试代码:
@Test
public void fun1() {
//1 创建容器对象--注意xml文件路径是相对于src下的
ApplicationContext ac=new ClassPathXmlApplicationContext("cn/ieate/applicationContext.xml");
//2 向容器要 user对象
User u = (User) ac.getBean("user");
//3 打印user对象
System.out.println(u);
}
方式二:静态工厂创建
&概述:调用UserFactory类的静态方法createUser方法创建名为user2的对象,放入容器中
&UserFactroy类代码
public class UserFactory {
public static User createUser() {
System.out.println("静态工厂创建User");
return new User();
}
}
&配置文件书写
<bean name="user2" class="ct.dde.UserFactory" factory-method="createUser">
</bean>
&测试代码
@Test
public void fun2() {
//1 创建容器对象--注意xml文件路径是相对于src下的
ApplicationContext ac=new ClassPathXmlApplicationContext("cn/ithereate/applicationContext.xml");
//2 向容器要 user对象
User u = (User) ac.getBean("user2");
//3 打印user对象
System.out.println(u);
}
方式三:实例工厂创建
& 概述:调用UserFactory的非静态方法createUser2方法创建名为user3的对象,放入容器中
& UserFactroy类代码
public class UserFactory {
public User createUser2() {
System.out.println("实例工厂创建User");
return new User();
}
}
& 配置文件书写
<!--先实例化UserFactory -->
<bean name="userFactory" class="cn.itheima.b_create.UserFactory"></bean>
<!--再调用UserFactory的成员方法 -->
<bean name="user3" factory-bean="userFactory" factory-method="createUser2"></bean>
&测试代码
@Test
public void fun3() {
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/itheima/b_create/applicationContext.xml");
User u = (User) ac.getBean("user3");
System.out.println(u);
}
<6>Bean元素的scope属性
scope值如下:
1 singleton(默认):单例,对象只创建一个实例
示例:
<bean name="user4" class="cn.ita.bean.User" scope="singleton"></bean>
2 prototype:多例,对象创建多个实例,每次获得才会创建,且每次创建都是新对象 .
整合Struts2,ActionBean的配置必须要用多例prototype配置,因为每请求都会创建新的Action对象,struts2是多例的
<bean name="user5" class="cn.ita.bean.User" scope="prototype"></bean>
3 request:在web项目中,Spring创建一个对象,并把该对象存进request中.
4 session:在web项目中,Spring创建一个对象,并把该对象存进session中.
5 globalSession:
在web项目中,应用Porlet环境,若没有Porlet环境,那么globalSession就相当于Session域。
Portlet是基于Java的Web组件,Portlet容器处理request并生成动态内容。
<7>Bean的生命周期
1 Bean元素的init-method 属性作为Bean内对象初始化执行的方法。
2 Bena元素的destroy-method属性作为Bean内对象销毁执行的方法
销毁方法会在单例创建Bean的工厂关闭时执行。
示例如下
<bean name="user6" class="cn.itheima.bean.User" init-method="init" destroy-method="destroy">
</bean>
public class User {
private String name;
private Integer age;
//初始化方法
public void init() {
System.out.println("初始化方法----");
}
//销毁方法
public void destroy() {
System.out.println("销毁方法----");
}
}
<8>Spring的Bean的属性注入
1 通过有参构造方法的方式注入
属性说明
name属性:参数名
value属性:参数值
index属性:参数索引(设置参数的先后位置,从而选择不同的参数顺序的构造函数的配置)
type属性:设置该参数的 类型
示例一:
<bean name="person" class="cn.ma.bean.Person">
<constructor-arg name="name" value="flower" index="0"/>
<constructor-arg name="age" value="233" index="1"/>
</bean>
public class Person {
private String name;
private Integer age;
public Person() {super();}
//有参数构造。
public Person(String name, Integer age) {
super();
this.name = name;
this.age = age;
}
示例二:构造方法的参数为某个对象时
<bean name="car" class="cn.ia.bean.Car">
<property name="name" value="兰基博尼"></property>
<property name="color" value="×××"></property>
</bean>
<bean name="user3" class="cn.ima.bean.User">
<constructor-arg name="name" value="1111" index="0" type="java.lang.Integer"></constructor-arg>
<!--ref属性关联着对象-->
<constructor-arg name="c" ref="car" index="1"></constructor-arg>
</bean>
2 set方法的方式(需提供getter/setter方法)
<bean name="user" class="cn.iima.bean.User">
<!-- 为User对象的name,age和Car c属性注入值-->
<property name="name" value="flowerr"></property>
<property name="age" value="21"></property>
<property name="c" ref="car"></property>
</bean>
public class User {
private String name;
public String getName() {return name;}
public void setName(String name) {this.name = name;}
private Integer age;
public void setAge(Integer age) {this.age = age;}
public Integer getAge() {return age;}
private Car c;
public Car getC() {return c;}
public void setC(Car c) {this.c = c;}
}
3 名称空间p的属性注入
* 注入说明:需导入p名称空间 xmlns:p="http://www.springframework.org/schema/p"
* 格式 -- p:属性名 或 p:属性名-ref
* 示例:
<bean name="user4" class="cn.ima.bean.User" p:name="jack" p:age="20" p:c-ref="car">
</bean>
4 Spel(Spring Exepression Language)注入方式
*格式:name="yyy" value="#{xxx}"
*示例一:
<bean name="user" class="cn.itheima.bean.User" >
<property name="name" value="#{‘flower‘}"></property>
<property name="age" value="#{22}"></property>
</bean>
*示例二:从spring容器取值注入属性里面
下面相当于把 存进spring容器内名为user对象的name属性值
赋予到本user5对象的name属性中
<bean name="user5" class="cn.ima.bean.User" >
<property name="name" value="#{user.name}"></property>
<property name="age" value="#{user1.age}"></property>
<property name="c" ref="car"></property>
</bean>
*示例三:把另一个Bean类的属性注入属性里面。
<bean name="xxx" class="aa.bb.eeexx"></bean>
<bean name="user5" class="cn.it.bean.User" >
<property name="name" value="#{xxx.name}"></property>
<property name="age" value="#{xxx.age}"></property>
</bean>
5 复杂类型注入
<bean name="cb" class="cn.ithma.d_injection.CollectionBean">
<!-- 数组注入 :如果数组中只准备注入一个元素(值|对象),直接使用 value|ref 即可
<property name="arr" value="tom"></property> -->
<!--数组的多元素注入 -->
<property name="arr">
<array>
<value>tom</value>
<value>123</value>
数组中注入对象
<ref bean="user5"/>
</array>
</property>
<!-- 单列集合 List注入:如果集合中只准备注入一个元素(值|对象),直接使用 value|ref 即可
<property name="list" value="myList"></property> -->
<property name="list">
<list>
<value>sb</value>
<value>250</value>
<ref bean="user4"/>
</list>
</property>
<!-- 双列集合 map注入 -->
<property name="map">
<map>
<entry key="url" value="jdbc:mysql:///crm"></entry>
<entry key="user" value-ref="user3"></entry>
<entry key-ref="user1" value-ref="user3"></entry>
</map>
</property>
<!-- properties类型(配置文件类型)注入 -->
<property name="prop">
<props>
<prop key="driverClass">com.jdbc.mysql.Driver</prop>
<prop key="userName">root</prop>
<prop key="password">234</prop>
</props>
</property>
</bean>
ApplicationContext接口有两个实现类,
ClassPathXmlApplicationContext:加载类路径下(src下)的Spring配置文件
FileSystemXmlApplicationContext:加载本地磁盘下Spring的配置文件
BeanFactory:在调用getBean()时才生成类的实例
ApplicationContext:容器启动,加载applicationContext.xml时便会创建类的实例。
方式一:创建工厂时加载多个配置文件
示例:
ApplicationContext ac=
new ClassPathXmlApplicationContext("a/applicationContext.xml,b/applicationContext.xml");
方式二:在一个配置文件中包含另一个配置文件
<import resource="xx/applicationContext.xml"/>
(1)概述:按照原生的Spring的IOC模式管理对象,每次获取applicationContext.xml配置文件中的bean对象时
都需要书写下面的代码
ApplicationContext ac=new ClassPathXmlApplicationContext("cn/iction/applicationContext.xml");
User u = (User) ac.getBean("user");
这样就会导致每次都会创建一个工厂类,服务器端的资源就浪费了,一般情况下一个工程只有一个Spring的工厂类.
解决方案:
让spring工厂在服务器启动时创建好,将这个工厂放入到ServletContext域中,每次获取工厂时,
从ServletContext域中获取即可,这就需用到ServletContextListener监听器,
监听ServletContext的创建和销毁。
配置书写如下:
<!--配置监听器,让spring工厂随工程启动而启动 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
core --Spring的核心
context-- 对应Spring的上下文对象环境(applicationContext.xml配置文件)
bean ---对应 applicationContext.xml配置文件的 <bean>元素
spel---对应 <bean>元素中的语法
特点简记:四大组件层层包含 core>context>bean>spel
标签:应用程序开发 属性 通过 etag 套件 enc 加载 上下文 getname
原文地址:http://blog.51cto.com/14008076/2314447