`
20386053
  • 浏览: 433066 次
文章分类
社区版块
存档分类
最新评论

Java动态代理剖析(一)

 
阅读更多

Java中的动态代理机制是Java中一个重要的设计模式,以后会经常用到,今天我来跟大家一起剖析一下这个重要的模式。

一. 为什么要使用动态代理

代理模式分为静态代理和动态代理,我们之所以会有动态代理的出现,就是因为静态代理存在不足,比如静态代理中会有大量重复的类和代码。

而我们的动态代理可以通过LogHandler类来动态的创建代理类,避免了编写各个代理类及重复的代码。

二.动态代理的执行原理

我们先来看看一个动态代理的小实例,主要有一个客户端,一个LogHandler类,一个接口和一个接口实现类:

客户端:

	package com.bjpowernode.pattern;
	
	public class Client {
	
		public static void main(String[] args) {
			//动态代理
			LogHandler handler = new LogHandler();
			//动态实例化代理类
			UserManager userManager = (UserManager)handler.newProxyObject(new UserManagerImpl());
			userManager.test();
		}
	}


LogHandler类:

	package com.bjpowernode.pattern;
	
	import java.lang.reflect.InvocationHandler;
	import java.lang.reflect.Method;
	import java.lang.reflect.Proxy;
	
	public class LogHandler implements InvocationHandler {
	
		//目标对象
		private Object targetObject;
		
		public Object newProxyObject(Object targetObject) {
			this.targetObject = targetObject;
			return Proxy.newProxyInstance(targetObject.getClass().getClassLoader() , targetObject.getClass().getInterfaces(), this);
		}
		
		//类似于Servlet filter中的doFilter方法,是一个拦截器
		public Object invoke(Object proxy, Method method, Object[] args )
				throws Throwable {
			System.out.println("start-->> " + method.getName() );
			if (args != null) {
				for (int i=0; i<args.length; i++) {
					System.out.println(args[i]);
				}
			}
			Object ret = null;
			try {
				//调用目标对象的方法(相当于filter中的filterChain.doFilter)
				ret = method.invoke(this.targetObject, args);
				System.out.println("success-->>" + method.getName());
			} catch(RuntimeException e) {
				System.out.println("error-->>" + method.getName());
				throw e;
			}
			return ret;
		}
	}


接口类

	package com.bjpowernode.pattern;
	
	public interface UserManager {
	
		public void test();
	}

实现类

	package com.bjpowernode.pattern;
	
	public class UserManagerImpl implements UserManager {
	
		@Override
		public void test() {
			System.out.println("UserManagerImpl.test()");
		}
	}


它的一个调用过程我已经在上一篇博客中提到了,主要是两条线:

1.动态实例化一个代理类

调用LogHandler的newProxyObject方法

newProxyObject调用Proxy.newProxyInstance方法

2.调用方法

调用代理类的方法userManager.test();

触发先调用LogHandler中的invoke方法;

LogHandler中的 invoke方法再调用Method的invoke方法;

Method中的invoke方法在调用UserManagerImpl的test()方法;

下面我对其中的几个方法进行一下讲解:

1. Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);

这个方法有三个参数,第一个参数targetObject.getClass().getClassLoader()是对目标类(即UserManagerImpl类)进行加载;知道这个代理类是对谁的代理。

第二个参数targetObject.getClass().getInterfaces()是对目标类的接口进行明确,知道代理类的职责,它需要代理的都有什么方法(代理只能是针对接口的)。

第三个参数this(即LogHandler类)是为了之后对其invoke方法的调用而做准备的。

2.invoke(Object proxy, Methodmethod, Object[] args )

这个方法也有三个参数,第一个参数proxy就是代理类(对于本例来说就是实例化的动态代理类userManager)。

第二个参数method就是代理类调用的方法。

第三个参数args就是方法的参数,如果方法没有参数即为null。

3.method.invoke(this.targetObject,args);

这个方法主要就是调用真实类的方法;第一个参数targetObject即为真实类。

第二个参数即为该方法所需参数。

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics