转载于 李建的博客 http://www.lijian4net.work/article/68
spring优势
1、方便解耦、简化开发
2、AOP编程的支持
3、声明式事物的支持
4、方便程序的测试
5、方便集成各种优秀的框架
6、降低JavaEE API的使用难度
7、源码是经典学习范例
程序的耦合
耦合:程序间的依赖关系
包括:
类之间的依赖
方法间的依赖
解耦:
降低程序间的依赖关系
实际开发过程中:
编译器不依赖,运行是才依赖
Bean:在计算机英语中,有可重用组件的含义
JavaBean:用Java语言编写的可重用组件
spring的IOC核心容器
ApplicationContext的三个常用实现类:
ClassPathXmlApplicationContext:它可以加载类路径下的配置文件,要求配置文件必须在类路径下,不在的话,加载不了
FileSystemXmlApplicationContext:它可以加载磁盘任意路径下的配置文件(必须有访问权限)
AnnotationConfigApplicationContext:它是用于读取注解创建容器的
核心容器的两个接口:
ApplicationContext: 单例对象适用
它在创建核心容器时,创建对象的策略是采用立即加载的方式,只要一读取完配置文件马上就创建配置文件中配置的对象
BeanFactory: 多例对象适用
它在创建核心容器时,创建对象的策略是采用延迟加载的方法,什么时候根据id获取对象了,什么时候才真正的创建对象
1、基于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"
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">
<!-- 标签内容 -->
</beans>
其中 beans后面的为spring约束,标签内容为spring配置内容
1.1 创建bean的三种方式
a、第一种方式:使用默认构造函数创建,在spring的配置文件中使用bean标签,配以id和class属性之后,没有其他属性和标签时,采用的就是默认构造函数创建bean对象,此时如果勒种没有默认构造函数,则对象无法创建
<bean id="userDao" class="work.lijian4net.www.dao.impl.UserDaoImpl"></bean>
b、第二种方式:使用普通工厂中的方法创建对象(使用某个类中的方法创建对象,并存入spring容器
<bean id="instanceFactory" class="work.lijian4net.www.factory.InstanceFactory"></bean>
<bean id="userDao" factory-bean="instanceFactory" factory-method="getUserDao"></bean>
public class InstanceFactory {
public IUserDao getUserDao(){
return new UserDaoImpl();
}
}
c、 第三种方法:使用工厂中的静态方法创建对象对象(使用某个勒种的静态方法创建对象,并存入spring容器
<bean id="userDao" class="work.lijian4net.www.factory.InstanceFactory" factory-method="getUserDao"></bean>
public class InstanceFactory {
public static IUserDao getUserDao(){
return new UserDaoImpl();
}
}
1.2 bean对象的作用范围
scope属性:
作用:用于指定bean的作用范围
取值:常用单利和多例的
singleton:单例的(默认)
prototype:多例的
request:作用于web应用的请求范围
session:作用于web应用的回话范围
global-session:作用于集群环境的回话范围(全局回话范围),当不是集群环境时,它就是session
1.3 bean对象的声明周期
单例对象
出生:当容器创建时对象出生
活着:只要容器还在,对象一直活着
死亡:容器销毁,对象消亡
总结:单例对象的声明周期和容器相同
多例对象:
出生:当我们使用对象时,spring框架创建对象
活着:对象使用过着中,一直活着
死亡:当对象尝试不用,且没有别的对象引用时,Java垃圾回收器回收
1.4 spring中的依赖
依赖注入:Dependency Injection
IOC的作用:
降低程序间的耦合(依赖关系)
依赖关系的管理以后都交给spring来维护
在当前类需要使用到其他类的对象,有spring为我们提供,我们只需要在配置文件中说明
依赖关系的维护:
就称之为依赖注入
能注入的数据:三类
基本类型和String
其他bean类型(在配置文件中或者注解配置过的bean)
复杂类型/集合类型
注入的方式:三种
使用构造函数提供
使用set方法提供
使用注解提供
1.4.1 构造函数注入
使用的标签:constructor-arg
标签出现的位置:bean标签的内部
标签中的属性
type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
index:用于指定要注册的数据给构造函数中指定索引位置的参数赋值,参数索引位置从0开始
name:用于指定给构造函数中指定名称的参数赋值(常用)
==================以上三个用于指定给哪个参数赋值==============
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型的数据,指的是在spring的ioc核心容器中出现过的bean对象
优势:
在获取bean对象时,注入数据是必须的的操作,否则对象无法创建
弊端:
改变了bean对象的实例化方式,使我们在创建对象时,如果用不到这些数据,也必须提供
<bean id="userService1" class="work.lijian4net.www.service.impl.UserServiceImpl">
<constructor-arg type="java.lang.String" value="test"></constructor-arg>
<constructor-arg name="age" value="12"></constructor-arg>
<constructor-arg index="2" ref="now"></constructor-arg>
</bean>
<!--配置一个日期对象-->
<bean id="now" class="java.util.Date"></bean>
public class UserServiceImpl implements IUserService {
private IUserDao userDao = new UserDaoImpl();
//对于经常变化的数据,不适用与注入的方式
private String name;
private Integer age;
private Date birthday;
public UserServiceImpl(String name,Integer age,Date birthday){
this.name= name;
this.age = age;
this.birthday = birthday;
}
public UserServiceImpl(){
}
@Override
public boolean Save(User user) {
try{
return userDao.insert(user)>0;
}catch (Exception ex){
System.out.println(ex.getMessage());
return false;
}
}
}
1.4.2 set方法注入 更常用
涉及的标签:property
出现的位置:bean标签的内部
标签的属性:
name:用于指定注入式所调用的set方法名称
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型的数据,指的是在spring的ioc核心容器中出现过的bean对象
优势:
创建对象时没有明确的限制,可以直接使用默认构造函数
弊端:
如果有某个成员必须又称,则获取对象时有可能set方法没有执行
<bean id="userService2" class="work.lijian4net.www.service.impl.UserServiceImpl2">
<property name="name" value="test"></property>
<property name="age" value="12"></property>
<property name="birthday" ref="now"></property>
</bean>
public class UserServiceImpl2 implements IUserService {
private IUserDao userDao = new UserDaoImpl();
//对于经常变化的数据,不适用与注入的方式
private String name;
private Integer age;
private Date birthday;
public void setName(String name) {
this.name = name;
}
public void setAge(Integer age) {
this.age = age;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
@Override
public boolean Save(User user) {
try{
return userDao.insert(user)>0;
}catch (Exception ex){
System.out.println(ex.getMessage());
return false;
}
}
}
1.4.3 复杂类型/集合类型注入
用于给List结构集合注入的标签:
list array set
用于给Map结构集合注入的标签:
map props
结构相同,标签可以互换
<bean id="userService3" class="work.lijian4net.www.service.impl.UserServiceImpl3">
<property name="myStrs">
<array>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</array>
</property>
<property name="myList">
<list>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</list>
</property>
<property name="mySet">
<set>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</set>
</property>
<property name="myMap">
<map>
<entry key="aaa" value="aaaaa"></entry>
<entry key="bbb">
<value>bbbbbbbb</value>
</entry>
</map>
</property>
<property name="myProps">
<props>
<prop key="aaa">cacac</prop>
<prop key="bbb">cbcbc</prop>
</props>
</property>
</bean>
public class UserServiceImpl3 implements IUserService {
private String[] myStrs;
private List<String> myList;
private Set<String> mySet;
private Map<String,String> myMap;
private Properties myProps;
public void setMyStrs(String[] myStrs) {
this.myStrs = myStrs;
}
public void setMyList(List<String> myList) {
this.myList = myList;
}
public void setMySet(Set<String> mySet) {
this.mySet = mySet;
}
public void setMyMap(Map<String, String> myMap) {
this.myMap = myMap;
}
public void setMyProps(Properties myProps) {
this.myProps = myProps;
}
@Override
public boolean Save(User user) {
System.out.println(Arrays.toString(myStrs));
System.out.println(myList);
System.out.println(mySet);
System.out.println(myMap);
System.out.println(myProps);
return false;
}
}
2 基于注解的配置
曾经的XML配置
<bean id="userDao" class="work.lijian4net.www.factory.InstanceFactory"
scope="" init-method="" destroy-method="">
<property name="" value="" / ref=""></property>
</bean>
2.1 用于创建对象的
他们的作用和在xml配置文件中编写一个<bean></bean>标签实现的功能是一样的
Component:
作用:用于把当前类存入spring容器中
属性:
value:用于指定bean的id,当我们不写时,默认是当前类名,且首字母改小写
Controller:一般用于表现层
Serivce:一般用于业务层
Respository:一般用于用在持久层
以上三个注解的作用与Component是一模一样的,他们三个是spring框架为我们提供明确的三层使用的注解,使我们的三层对象更加清晰
2.2 用于注入数据的
他们的作用就和在xml配置文件中的bean标签中写一个<property></property>标签的作用是一样的
Autowired
作用:自动按照类型注入,只要容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功
如果IOC容器中没有任何bean的类型和要注入的变量类型匹配,则报错。
如果IOC容器中有多个类型匹配时:
出现位置:
可以是变量上,也可以是方法上
细节:在使用注解注入时,set方法不是必须的了
Qualifier:
作用:在按照类中注入的基础之上再按照名称注入。它在给类成员注入时不能单独使用。但是在给方法参数注入时可以
属性:
value:用于指定注入bean的id
Resource
作用:直接安好bean的id注入。它可以独立使用
属性:
name:用于指定bean的id
以上三个注入都只能注入其他bean类型的数据,而基本类型和String类型无法使用上述注解实现
另外,集合类型的注入只能通过xml来实现
Value
作用:用于植入基本类型和String类型的数据
属性:
value:用于指定数据的值,它可以使用spring中的SpEL(spring的el表达式)
SpEL的写法:${表达式}
2.3 用于改变作用范围的
他们的作用就和在bean标签中使用scope标签的作用是一样的
Scope
作用:用于指定bean的作用范围
属性:
value:指定范围的取值。常用取值:singleton prototype
和声明周期相关(了解)
他们的作用就和在bean标签中使用init-method和destory-method标签的作用一样
PreDestory
作用:用于指定销毁方法
PostConstruct
作用:用于指定初始化方法
@Repository(value = "annoDao")
public class AnnoDaoImpl implements IAnnoDao {
@Override
public void save(User user) {
}
}