Spring的两个核心特性:
- 依赖注入(dependency injection,DI)
- 面向切面编程(aspect oriented programming,AOP)
依赖注入(dependency injection,DI)
- 没有使用依赖注入时:以前每个POJO(Plain Ordinary Java Object,简单的Java对象)在创建的时候会主动的去获取依赖。从代码上的体现是一个类中有实例化另一个类的对象(耦合度高)。我们需要将相互协作的对象引用赋给它。
- 三种依赖注入方式:setter方法注入,构造器注入,接口注入(就是在接口中声明一个方法,参数是依赖的对象)。
- 依赖注入作用:将各个相互协作的模块代码保持松散耦合。
- 下面我们通过几个简单的类来了解一下依赖注入:
先声明两个接口People与Fruit。
public interface People {// people接口
void eat();
}
public interface Fruit {// 水果接口
void speak();
}
接下来是分别实现他们的类Watermelon与Student。
public class Watermelon implements Fruit{// 西瓜类
@Override
public void speak() {
System.out.println("我是西瓜,我要被吃掉了!");
}
}
public class Student implements People {// 学生类
private Fruit fruit;
public Student(Fruit fruit) {// 构造器注入依赖的对象
this.fruit = fruit;
}
@Override
public void eat() {
fruit.speak();
}
}
到这里我们可以看出Student类依赖Watermelon类。当调用Student的eat方法时需要调用Watermelon的speak方法。
我们接下来用XML对这两个应用组件进行装配(wiring)。
<?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-->
<bean id="fruit" class="com.qsh.springaction.Watermelon"/>
<!--人声明为Spring的bean-->
<bean id="people" class="com.qsh.springaction.Student">
<constructor-arg ref="fruit"/>
</bean>
</beans>
最后我们进行测试。用spring上下文全权负责对象的创建和组装。
public class TestEat {
public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("eat.xml");
People people = context.getBean(People.class);
people.eat();
context.close();
}
}
测试结果。
面向切面编程(aspect oriented programming,AOP)
- 是什么:促使软件系统实现关注点分离的一项技术。系统由许多不同组件构成,每个组件负责一块特定功能。除了实现自身核心功能外,这些组件还承担着额外职责,如日志,事务管理和安全这样的系统服务,这样的系统服务被称为横切关注点。因为他们会横跨多个组件。
- 作用:有助于横切关注点与它们所影响的对象之间的解耦。
- 个人理解:每个组件除了实现自身的功能,还需要实现其他的如事物管理等功能(称为横切关注点)。我们将这功能单独抽取出来成一个模块,每个组件在工作的过程中,这个模块神不知鬼不觉的为每个组件实现额外功能。
- 个人理解的切面图:应用组件:实现各自功能的代码。横切关注点:每个组件的额外业务,相同的代码。切面:将横切关注点模块化出来的一个类。连接点:应用执行时能够插入切面的点(超级多)切点:匹配其中一个或多个连接点。
- 通过几段代码讲解AOP:
我们还是有上面的Student 和 Watermelon类。需要新增加一个Action类,就是一会的切面。
public class Action {// 行为类
public void beforeEat() {
System.out.println("吃前拿刀,嚓嚓嚓");
}
public void afterEat() {
System.out.println("吃后洗手,哗哗哗");
}
}
需要在XML中加入Spring AOP命名空间,将Action方法声明为Spring的bean,然后切面引用这个bean。切点为student的eat方法,在切点前后加入前置通知和后置通知。
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:aop="http://www.springframework.org/schema/aop"
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
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.2.xsd">
<!--水果声明为Spring的bean-->
<bean id="fruit" class="com.qsh.springaction.Watermelon"/>
<!--人声明为Spring的bean-->
<bean id="people" class="com.qsh.springaction.Student">
<constructor-arg ref="fruit"/>
</bean>
<!--行为声明为Spring的bean-->
<bean id="action" class="com.qsh.springaction.Action"/>
<!--用spring aop的命名空间把Action声明为一个切面-->
<aop:config>
<!--引用Action的bean-->
<aop:aspect ref="action">
<!--声明Student的eat方法为一个切点-->
<aop:pointcut id="eating" expression="execution(* *.eat(..))"/>
<!--前置通知,在调用eat方法前调用Action的beforeEat方法-->
<aop:before pointcut-ref="eating" method="beforeEat"/>
<!--后置通知,在调用eat方法后调用Action的afterEat方法-->
<aop:after pointcut-ref="eating" method="afterEat"/>
</aop:aspect>
</aop:config>
</beans>
运行结果为: