【spring框架】利用动态代理实现AOP

目标:生成一个UserDao的代理,然后在代理执行的过程中,把关于日志记录的逻辑加进去。

UserDao.java:
package cn.edu.hpu.dao;

import cn.edu.hpu.model.User;
public interface UserDao {
	public void save(User u);
}

UserDaoImpl.java:
package cn.edu.hpu.dao.Impl;

import cn.edu.hpu.dao.UserDao;
import cn.edu.hpu.model.User;


public class UserDaoImpl implements UserDao{


	public void save(User u) {
	    System.out.println("add success!!");		
	}


}

现在需要要在执行UserDaoImpl的save方法之前和之后执行LogIntercept类的beforeMethod与afterMethod方法赖创建日志。


首先让LogIntercept类实现动态代理的接口InvocationHandler
然后重写invoke方法。
package cn.edu.hpu.aop;


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;


//日志的拦截器
public class LogIntercept implements InvocationHandler{
	private Object terget;
	
	public void beforeMethod(){
		System.out.println("save start...");
	}
	
	public void afterMethod(){
		System.out.println("save end...");
	}


	public void setTarget(Object terget) {
		this.terget=terget;
	}
	//无论调用代理对象的哪个方法,invoke方法都要被调用
	public Object invoke(Object proxy, Method m, Object[] args)
			throws Throwable {
		beforeMethod();
		m.invoke(terget, args);
		afterMethod();
		return null;
	}
}

测试方法:
@Test
public void testProxy(){
	//产生一个被代理的对象
	UserDao userDao=new UserDaoImpl();
	LogIntercept li=new LogIntercept();
	li.setTarget(userDao);
	UserDao userDaoProxy=(UserDao)Proxy.newProxyInstance(
			userDao.getClass().getClassLoader(), 
			new Class[]{UserDao.class}, 
			li);
	userDaoProxy.save(new User());
}

测试结果:
save start...
add success!!
save end...


当你有一件事情,需要在多个方法上面都要加相同的业务逻辑的时候,用动态代理会帮你省好多事情。

剖析一下动态代理的实现:
UserDao userDaoProxy=(UserDao)Proxy.newProxyInstance(
			userDao.getClass().getClassLoader(), 
			new Class[]{UserDao.class}, 
			li);
得到的是UserDao接口吗?测试一下:
System.out.println(userDaoProxy.getClass());
得到:class $Proxy4
说明userDaoProxy并不是UserDao接口,而是JDK的一个动态代理类$Proxy4,它是根据你传给它的参数中的接口那一项,得到接口本身,继而拿到接口的所有方法,哪个方法调用都会把接口本身传给invocationHandlder(li的invoke方法拿到)。

好了,我们写完了动态代理的AOP,但是,到底什么是AOP(Aspect-Oriented-Programming)面向切面编程呢?

你会发现,我们正常的程序写起来就是一条直线,比如:
接受用户名密码--->连接数据库--->创建XXX对象--->存储--->创建日志--->END

而面向切面编程就好像我们在中间"咔嚓"砍掉一刀,在上面加上逻辑,在另外一个地方"咔嚓"一刀,也在上面加逻辑,而方法自己本身并不知道。

面向切面编程是面向对象编程的有一伟大思想。应用的地方太多,权限、日志、检查、事务(可以在CRUD方法前加上Transiention.start(),事务结束后加上Transiention.commit())、异常的管理(单独写一个业务逻辑,想给哪个方法加异常管理就加)。

比如权限,再给每个JSP文件设权限的时候,在每个JSP文件开始的时候要做个检查,正常的写JSP文件或方法,我们的业务逻辑才是我们的主要内容,但是由于要检查权限,非要在前面加上检查权限的代码,代码会很繁琐,不会专注于特定的内容了,而且去掉它的时候会很麻烦。

但是利用面向切面的编程,就不会产生这些问题,而且这些加上的业务逻辑可以利用Spring的配置文件轻松的去掉或者加上。

现在是实现了InvocationHandler,如果利用Spring,不用实现InvocationHandler{接口也可以,它就是使用了直接生成二进制码,用继承。
Sturts2中的拦截器使用的就是AOP切面编程

顺便说一下,一般写程序推荐面向接口编程。

转载请注明出处:http://blog.csdn.net/acmman/article/details/44276427

你可能感兴趣的:(动态代理,spring,AOP,javaee,面向切面编程)