Spring AOP入门 (二)动态代理

动态代理指的是通过一个代理对象来创建需要的业务对象,然后在这个代理对象中统一进行各种需求的处理。

1. 写一个类实现相应的InvocationHandler接口。

public class LogProxy implements InvocationHandler {
	
	//2. 创建一个代理对象. 由于不知道代理的对象类型,所以这里用Object
	private Object target;
	
	private LogProxy(){}
	
	//3. 创建一个方法来生成对象,这个方法的参数是要代理的对象
	// getInstance所返回的对象就是代理对象。
	public static Object getInstance(Object o){
	    	//3.1 创建LogProxy对象
			LogProxy proxy=new LogProxy();
			
			//3.2 设置这个代理对象
			proxy.target=o;
			
			//3.3 通过Proxy的方法创建代理对象,第一个参数是要代理对象的classloader。
			// 第二个参数是代理对象实现的所有接口。
			// 第三个参数是实现类InvocationHandler的对象。
			// 此时的result就是一个代理对象,代理的是o
			Object result=Proxy.newProxyInstance(o.getClass().getClassLoader(), o.getClass().getInterfaces(), proxy);
			return result;
	}
	
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		Logger.info("进行了相应的操作");
		System.out.println("method name= "+method.getName());
		Object obj=method.invoke(target, args);
		return obj;
	}

}



2. 在beans.xml中配置一个动态dao:

 <!-- 创建一个动态dao,通过LogProxy实现 -->      <!-- 通过factory-method来指定创建对象的静态方法 -->
         <bean id="userDynamicDao" class="com.lj.proxy.LogProxy" factory-method="getInstance">
         	<constructor-arg ref="userDao"/>  
         	<!-- 这里的userDao就是com.lj.dao.UserDao里的@Repository("userDao") -->
         </bean>


这里相当于通过LogProxy里面的方法创建了一个IUserDao userDynamicDao.
此时我们在service层就不再使用userDao,而是userDynamicDao.

3. 修改UserService.java:
@Resource(name="userDynamicDao")
	private IUserDao userDao;


通过以上的操作,在UserService中的任何一个userDao的操作,都会被记录下来。

======================================================
我们还可以通过Annotatioin类来决定方法是否被记录。

@Retention(RetentionPolicy.RUNTIME)
public @interface LogInfo {
	public String value() default "";
}


再找到IUserDao, 加入注释:

@LogInfo("所有用户都被查询了 by @LogInfo")
	public List<User> list();



最后修改代理类LogProxy:
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		//Logger.info("进行了相应的操作");
		if(method.isAnnotationPresent(LogInfo.class)){
			System.out.println(method.getAnnotation(LogInfo.class).value());
		}
		
		System.out.println("method name= "+method.getName());
		Object obj=method.invoke(target, args);
		return obj;
	}


最后执行list方法,便会看到
所有用户都被查询了 by @LogInfo
method name= list


猜你喜欢

转载自alleni123.iteye.com/blog/1987473