欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Spring AOP上篇

程序员文章站 2022-04-25 21:27:16
...
[size=x-large]
在Spring中、AOP的编程思想实际上跟我们之前用的过滤器或拦截器实现思路差不多;在了解AOP之前、我们先来说下代理模式吧;
代理模式:当程序不能或不想处理某个任务时,可以通过引用代理类来帮助它实行该任务。
目标对象 -- 委托人即被代理的对象
代理对象 -- 不能违背委托人的真实意愿,
代理要求:(代理对象必须跟目标对象具有相同的抽象接口)通俗讲就是不能违背目标对象的本意;所以、这里我们就要引入一个抽象接口的角色。

下面看例子:
[/size]


/**
* 代理模式测试
*/
public void testProxy1()throws Exception{
  //同样的、先解析spring配置文件,通过解析实例化相应对象
ApplicationContext ioc= new ClassPathXmlApplicationContext
("applicationContext.xml");
//这里一样不能实例化得到对象、要通过getBeam方法获取已实例化的对象
TestAction action = (TestAction)ioc.getBean("testAction");
try {
  //然后再调用要实现的方法
action.saveObject();

action.findByName();
} catch (Exception e) {
e.printStackTrace();
}
}
在spring配置文件中、我们需要配置如下方法;
<!-- 代理类 此配置旨在表达所找的代理对象-->
<bean id="proxyDAO" class="hn.spring.why.ProxLogDAO">
<property name="baseDAO" ref="hibernateDAO"/>
</bean>
<!-- 此配置是原来的action所依赖的DAO 实现类、但现在通过上面的配置,引入到代理类中去实现了-->
<bean id="testAction" class="hn.spring.why.TestAction"
scope="prototype">
<!-- 维护当前action类依赖的DAO实现类-->
<property name="baseDAO" ref="proxyDAO"/>

</bean>

接着代理目标对象输出日志信息
public class ProxLogDAO implements IBaseDAO {

private IBaseDAO baseDAO; //代理的目标对象
  //注入目标对象
public void setBaseDAO(IBaseDAO baseDAO) {
this.baseDAO = baseDAO;
}

public void saveObject(Object obj) {
System.out.println("【saveObject 日志处理 Begin............】");
baseDAO.saveObject(obj); //调用目标对象上的目标方法
System.out.println("【saveObject 日志处理 end............】");
}

public List getAllObectsByName(String name) {
System.out.println("【getAllObectsByName 日志处理 Begin............】");
List list = baseDAO.getAllObectsByName(name);
System.out.println("【getAllObectsByName 日志处理 end............】");
return list;
}

}


[size=x-large]

这样就起到了代理作用、但这中静态代理也有问题,就是它必须基于抽象接口提供的不同代理类。系统中有多少抽象接口、就需要写多少代理类;它的灵活度是不够的。所以、我们大都实际使用中大都用的是动态代理。

动态代理:基于java发射API
在javaJDK文档中、我们要熟悉的有两个类,一个Proxy、它是提供用于创建动态代理类和静态方法;即动态地创建和代理类及代理类的实例。类中创建实例的方法是newProxyInstance();这个代理类实际上不存在的、它是运行时动态产生的也就是虚拟的。但是它是关联的呢、实际上,每个代理实例都有一个关联的调用处理 程序对象,它可以实现接口InvocationHandler;而在这个接口中有一个invoke方法、只要你调用这个接口,就会实现这个方法。、

下面来看例子吧
[/size]


首先我们创建一个动态代理类、让它实现接口InvocationHandler

public class DynamicProxyDAO implements InvocationHandler {

//同时内部提供invoke方法
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {

在这个类中、除了提供的方法,我们需要自己提供一个方法来实现动态创建代理类
public Object createProxyInstance(){
/**
* @param loader 用于加载代理类动态实现的接口
* @param interfaces 目标对象上的所实现的接口列表(代理对象必须与目标对象具备相同的抽象接口)
* @param this 代理对象绑定的处理程序(代理对象上的方法被调用时,就会调用该接口中的invoke()
*/

return Proxy.newProxyInstance(loader ,interfaces , this);
}

在实现此方法前、我们必须给它提供一个目标对象
private Object target; //被代理的目标对象
//用set方法来注入
public void setTarget(Object target){
this.target = target;
}
然后在创建动态代理方法中、我获取目标对象的类信息
Class clazz = target.getClass();
再将返回的值变换
return Proxy.newProxyInstance(clazz.getClassLoader(),
clazz.getInterfaces(), this);


最后、同样地需要在spring配置文件中配置bean文件
<!-- 动态代理 -->
<bean id="dynamicDAO"class="hn.spring.why.DynamicProxyDAO">
<property name="target" ref="hibernateDAO"/>
</bean>

现在我们来调用动态创建代理的方法
public void testProxy2()throws Exception{
ApplicationContext ioc = new ClassPathXmlApplicationContext
("applicationContext.xml");
DynamicProxyDAO proxyDAO = (DynamicProxyDAO)ioc.getBean("dynamicDAO");

//创建代理类的实例
IBaseDAO baseDAO = (IBaseDAO)proxyDAO.createProxyInstance();

//调用代理对象上的代理方法
baseDAO.saveObject("");
}

[size=x-large]

到这里、我似乎认为基本已经可以实现了;但是测试时发现这里还有一个重要的地方没做:反射调用
在上面的invoke方法中、我们代理方法中没调用目标对象上的目标方法;
Object result = method.invoke(target, args);
这里的args为参数数组
Return result;
这样便可以通过动态代理来调用实现类中方法;这就是动态代理。
[/size]
相关标签: aop