Spring中的AOP面向切面编程和Spring对JDBC的支持

一、Spring-AOP

AOP面向切面编程(Aspect-Oriented Programming),是对传统OOP的补充。AOP使用是使用动态代理实现的。动态代理我们已经非常熟悉了。我也将逐渐的由实现原理转向应用,因为 我们在学习JavaWEB基础的时候,原理已经学习过了。

 

AspectJ是Java社区里最完整最流行的AOP框架,Spring2.X使用的正是这个框架。AspectJ已经被广泛应用,她具有大好前 程。

 

AspectJ支持注解和XML配置。

 

1.启用AspectJ注解支持

1).AspectJ需要两个Jar包:spring-framework-2.5.6.SEC01\lib\aspectj\目录下 “aspectjrt.jar”和“aspectjweaver.jar”。

2).在Bean配置文件中添加“ < aop:aspectj-autoproxy />

 

 

2.使用AspectJ注解声明切面

AspectJ 支持 5 种类型的通知注解:

1) @Before: 前置通知, 在方法执行之前执行。

2) @After: 后置通知, 在方法执行之后执行 。

3) @AfterRunning: 返回通知, 在方法返回结果之后执行。

4) @AfterThrowing: 异常通知, 在方法抛出异常之后。

5) @Around: 环绕通知, 围绕着方法执行。

 

我们来看个简单的例子,算术计算器。

1).定义一个接口:

package cn.itcast.cc.spring.aspectj;

public interface ArithmeticCalculator {

void add(int i, int j);// 无返回值,用于测试

void sub(int i, int j);// 无返回值,用于测试

int mul(int i, int j);

int div(int i, int j);

}
 

2) .实现类:

package cn.itcast.cc.spring.aspectj;

import org.springframework.stereotype.Component;

@Component(value="calc")

public class ArithmeticCalculatorImpl implements ArithmeticCalculator {

@Override

public void add(int i, int j) {

int result = i + j;

System.out.println(result);

}

@Override

public void sub(int i, int j) {

int result = i - j;

System.out.println(result);

}

@Override

public int div(int i, int j) {

int result = i / j;

return result;

}

@Override

public int mul(int i, int j) {

int result = i * j;

return result;

}

}
 

3) .AspectJ切面类:

package cn.itcast.cc.spring.aspectj;

import org.aspectj.lang.*;

import org.aspectj.lang.annotation.*;

import org.springframework.stereotype.Component;

@Aspect // AspectJ切面Bean

@Component // Bean

public class ArithmeticCalculatorAspect {

// 前置通知

@Before(value = "execution (* cn.itcast.cc.spring.aspectj.ArithmeticCalculator.* (..))")

public void beforeMethodLoggin(JoinPoint jp) {

System.out.println("before " + jp.getSignature().getName());

}

// 后置通知

@After(value = "execution (* cn.itcast.cc.spring.aspectj.ArithmeticCalculator.* (..))")

public void afterMethodLoggin(JoinPoint jp) {

System.out.println("after " + jp.getSignature().getName());

}

// 返回通知

@AfterReturning(value = "pointCut()", returning = "result")

public void afterMethodLogginReturn(JoinPoint jp, Object result) {

System.out.println("after " + jp.getSignature().getName() + " return " + result);

}

// 异常通知

@AfterThrowing(value = "pointCut()", throwing = "e")

public void errorMethodLoggin(JoinPoint jp, Throwable e) {

System.out.println("method " + jp.getSignature().getName()

+ " throwing " + e);

}

// 环绕通知,通过它可以实现上面所有的通知。

@Around("execution (* cn.itcast.cc.spring.aspectj.ArithmeticCalculator.* (..))")

public Object aroundMethodLoggin(ProceedingJoinPoint pjp) {

System.out.println("around_before " + pjp.getSignature().getName());

Object result = null;

try {

result = pjp.proceed();

} catch (Throwable e) {

e.printStackTrace();

System.out.println("around_error " + pjp.getSignature().getName());

}

System.out.println("around_after " + pjp.getSignature().getName());

System.out.println("around_after " + pjp.getSignature().getName()

+ " return " + result);

return result;

}

// 重用切入点,其他 “通知方法”可以引用此方法名称“pointCut()”。

@Pointcut(value = "execution (* cn.itcast.cc.spring.aspectj.ArithmeticCalculator.* (..))")

public void pointCut() {

}

}
 

4) .Bean配置文件:

<?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:aop="http://www.springframework.org/schema/aop"

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/aop http://www.springframework.org/schema/aop/spring-aop.xsd

http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

<context:component-scan base-package="cn.itcast.cc.spring.aspectj" />

<aop:aspectj-autoproxy />

</beans>
 

5).测试类:

package cn.itcast.cc.spring.aspectj;

import org.springframework.context.ApplicationContext;

import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Main {

public static void main(String[] args) {

ApplicationContext ac = new ClassPathXmlApplicationContext("beans-aspect.xml");

ArithmeticCalculator calc = (ArithmeticCalculator) ac.getBean("calc");

calc.add(1, 2);

// calc.sub(2, 1);

// calc.mul(3, 3);

// calc.div(10, 2);

}

}
 

上面是较为常用的注解。

 

2.使用XML文件声明切面

使用XML文件声明切面没有使用注解声明切面直观,简单。

1) .删除所有AspectJ注解

2) .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:aop="http://www.springframework.org/schema/aop"

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/aop http://www.springframework.org/schema/aop/spring-aop.xsd

http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

<bean id="arithmeticCalculatorAspect" class="cn.itcast.cc.spring.xml.ArithmeticCalculatorAspect" />

<bean id="arithmeticCalculatorImpl" class="cn.itcast.cc.spring.xml.ArithmeticCalculatorImpl" />

<aop:config>

<aop:pointcut

expression="execution(* cn.itcast.cc.spring.xml.ArithmeticCalculator.* (..))"

id="pointcut" />

<aop:aspect order="1" ref="arithmeticCalculatorAspect">

<aop:before method="beforeMethodLoggin" pointcut-ref="pointcut"/>

<aop:after method="afterMethodLoggin" pointcut-ref="pointcut"/>

<aop:after-returning method="afterMethodLogginReturn" returning="result" pointcut-ref="pointcut"/>

<aop:after-throwing method="errorMethodLoggin" throwing="e" pointcut-ref="pointcut"/>

<aop:around method="aroundMethodLoggin" pointcut-ref="pointcut"/>

</aop:aspect>

</aop:config>

</beans>
 

3.引入通知

这是一个高级应用。我们知道Java只能单继承,使用这个引入通知就能实现多集成! 这一点是我没想到的,但想一想Java的反射机制和动态代理实现这一点并不难。

 

我们为上边的计算器,扩展 两个应用:计算最大值和最小值,他们分别是两接口和对应的实现类“ M ax Calculator ”(Impl)和“ MinCalculator ”(Impl)

 

使用注解的方式实现引入通 知,在 ArithmeticCalculatorAspect .java中添加

@DeclareParents(value="cn.itcast.cc.spring.aspectj.ArithmeticCalculator*",

defaultImpl=MinCalculatorImpl.class)

public MinCalculator minCalculator;

@DeclareParents(value="cn.itcast.cc.spring.aspectj.ArithmeticCalculator*", defaultImpl=MaxCalculatorImpl.class)

public MaxCalculator maxCalculator;
 

使用XML文件实现引入通 知:

<aop:declare-parents types-matching="cn.itcast.cc.spring.xml.ArithmeticCalculator*" 

implement-interface="cn.itcast.cc.spring.xml.MaxCalculator"

default-impl="cn.itcast.cc.spring.xml.MaxCalculatorImpl"/>

<aop:declare-parents types-matching="cn.itcast.cc.spring.xml.ArithmeticCalculator*" 

implement-interface="cn.itcast.cc.spring.xml.MinCalculator"

default-impl="cn.itcast.cc.spring.xml.MinCalculatorImpl"/>
 

引入通知的使用:

 

ApplicationContext ctx = new ClassPathXmlApplicationContext("beans-xml.xml");

Object obj = ctx.getBean("calc");

ArithmeticCalculator arithmeticCalculator = (ArithmeticCalculator) obj;

// 可以转换

MinCalculator minCalculator = (MinCalculator) obj;

// 可以转换

MaxCalculator maxCalculator = (MaxCalculator) obj;
 

 

二、Spring-JDBC

JDBC中Spring中似乎比较常用,但使用起来非常简单。

 

1.引入数据源

昨天介绍的使用外部Bean正是引入C3P0数据源的方法,在此我们就不重复了。

 

2.使用方式

我们使用spirng为我们提供的一个CURD操作类——JdbcTemplate,在xml配置文件中添加一个Bean:

<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">

<property name="dataSource" ref="dataSource"/>

</bean>
 

在程序中使用:

package cn.itcast.cc.spring.jdbc;

import org.junit.Test;

import org.springframework.context.ApplicationContext;

import org.springframework.context.support.ClassPathXmlApplicationContext;

import org.springframework.jdbc.core.JdbcTemplate;

public class JdbcTest {

private ApplicationContext ac = new ClassPathXmlApplicationContext(

"beans.xml");

private JdbcTemplate jdbcTemplate = (JdbcTemplate) ac

.getBean("jdbcTemplate");

@Test

public void testInsert() {

String sql = "insert into customers(customer_name,home_address,mobile_phone) VALUES (?,?,?)";

Object[] args = { "changcheng", "DaLian", "1398639955" };

this.jdbcTemplate.update(sql, args);

}

}
 

在此我只列出了insert的方法,其他方法可以查看Spring手册,十分简单。

 

Spring还为我们提供了一个更简单的CURD类—— SimpleJdbcTemplate , 它使用起来比 JdbcTemplate简单些,在此也不多做介绍了。

你可能感兴趣的:(spring,AOP,编程,bean,jdbc)