Spring AOP上篇

[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]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值