java代理

代理分为静态代理和动态代理

静态代理:

通过聚合实现。即代理对象和被代理对象实现相同的接口。代理对象在重写方法时会调用被代理对象的方法。而在调用前后会加入相关逻辑,例如日志,安全检查等。

举例

接口(代理对象和被代理对象都要实现)

package com.fw.proxy;
public class Car implements Moveable{

	@Override
	public void run() {
		System.out.println("car moving...");
	}

}

Car对象(被代理对象)

package com.fw.proxy;
public class Car implements Moveable{

	@Override
	public void run() {
		System.out.println("car moving...");
	}

}

CarProxy对象(代理对象)

package com.fw.proxy;

public class CarLogProxy implements Moveable{
	
	/**
	 * 这是静态代理的实现。动态代理要实现的目的如下:
	 * 传入被代理对象的接口之后,由用户自定义好代理实现的内容,即可完成代理。
	 * 需要做的工作:
	 * 动态生成一个代理对象,这个对象对于用户来说是不可见的,就如同这里的CarLogProxy一样。只不过这里的CarLogProxy
	 * 是可见的。 
	 * 
	 * 
	 * 
	 */
	private Moveable m;
	public  CarLogProxy(Moveable m) {
		this.m = m;
	}

	@Override
	public void run() {
		System.out.println("日志开始");
		m.run();
		System.out.println("日志结束");
	}
}

以上是静态代理

 

动态代理的原理:

JDK里动态代理需要根据用户传入(接口信息)和自定义(对InvocationHandler的实现)的信息来动态编译生成一个代理类,拿上面的例子举例,JDK需要做的就是帮助用户自动生成CarLogProxy这个类和对象。那么JDK如何实现呢

第一步,由用户自定义实现InvocationHandler接口的类

package com.bjsxt.proxy;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class TimeHandler implements InvocationHandler{
	//该类的作用就是在调用被代理类的同时在其前后加入需要代理的逻辑,例如日志,安全检查等等

	private Object target;

	public TimeHandler(Object target) {
		super();
		this.target = target;
	}

	@Override
	public void invoke(Object o, Method m) {
		long start = System.currentTimeMillis();
		System.out.println("starttime:" + start);
		//System.out.println(o.getClass().getName());
		try {
			m.invoke(target);
		} catch (Exception e) {
			e.printStackTrace();
		}
		long end = System.currentTimeMillis();
		System.out.println("time:" + (end-start));
	}

}

 

第二步动态编译生成Proxy对象。该对象负责产生对用户不可见的代理对象,主要工作就是将字符串编译成class文件,load到内存之中再创建代理对象

package com.bjsxt.proxy;

import java.io.File;
import java.io.FileWriter;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;

import javax.tools.JavaCompiler;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import javax.tools.JavaCompiler.CompilationTask;

public class Proxy {
	public static Object newProxyInstance(Class infce, InvocationHandler h) throws Exception { //JDK6 Complier API, CGLib, ASM
		String methodStr = "";
		String rt = "\r\n";
		
		Method[] methods = infce.getMethods();
		/*
		for(Method m : methods) {
			methodStr += "@Override" + rt + 
						 "public void " + m.getName() + "() {" + rt +
						 	"   long start = System.currentTimeMillis();" + rt +
							"   System.out.println(\"starttime:\" + start);" + rt +
							"   t." + m.getName() + "();" + rt +
							"   long end = System.currentTimeMillis();" + rt +
							"   System.out.println(\"time:\" + (end-start));" + rt +
						 "}";
		}
		*/
		for(Method m : methods) {
			methodStr += "@Override" + rt + 
						 "public void " + m.getName() + "() {" + rt +
						 "    try {" + rt +
						 "    Method md = " + infce.getName() + ".class.getMethod(\"" + m.getName() + "\");" + rt +
						 "    h.invoke(this, md);" + rt +
						 "    }catch(Exception e) {e.printStackTrace();}" + rt +
						
						 "}";
		}
		
		String src = 
			"package com.bjsxt.proxy;" +  rt +
			"import java.lang.reflect.Method;" + rt +
			"public class $Proxy1 implements " + infce.getName() + "{" + rt +
			"    public $Proxy1(InvocationHandler h) {" + rt +
			"        this.h = h;" + rt +
			"    }" + rt +
			
			
			"    com.bjsxt.proxy.InvocationHandler h;" + rt +
							
			methodStr +
			"}";
		String fileName = 
			"d:/src/com/bjsxt/proxy/$Proxy1.java";
		File f = new File(fileName);
		FileWriter fw = new FileWriter(f);
		fw.write(src);
		fw.flush();
		fw.close();
		
		//compile
		JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
		StandardJavaFileManager fileMgr = compiler.getStandardFileManager(null, null, null);
		Iterable units = fileMgr.getJavaFileObjects(fileName);
		CompilationTask t = compiler.getTask(null, fileMgr, null, null, null, units);
		t.call();
		fileMgr.close();
		
		//load into memory and create an instance
		URL[] urls = new URL[] {new URL("file:/" + "d:/src/")};
		URLClassLoader ul = new URLClassLoader(urls);
		Class c = ul.loadClass("com.bjsxt.proxy.$Proxy1");
		System.out.println(c);
		
		Constructor ctr = c.getConstructor(InvocationHandler.class);
		Object m = ctr.newInstance(h);
		//m.move();

		return m;
	}
}

故通过InvocationHandler和Proxy类即可以实现动态代理

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值