一、什么是动态代理
在工作中,有一个实现好的类,后面我要给这个类的方法增加一些功能,但是我们不能直接在源代码上面修改(侵入式修改),因为改错了,整个项目可能会瘫痪。于是我们需要一个代理类做额外的增加的功能,在对应的位置通过反射获取源代码对应的功能
简单说就是,我有一个类叫做鸡哥,它有唱和跳的功能,现在给鸡哥增加准备话筒,场地收钱的额外功能,但是鸡哥只想唱和跳,于是通过中介公司请了一个代理,代理需要做的就是准备话筒场地收钱,至于唱和跳就通过反射,获取鸡哥的成员方法来使用。
Java通过什么来保证代理的样子?
通过接口保证,后面的对象和代理需要实现同一个接口(就像上面的中介公司)
接口中就是被代理的所有方法。
二、实现实例代码
要增加功能的类的代码
要给鸡哥唱歌增加准备话筒和收钱的功能
要给鸡哥跳舞增加准备场地和收钱的功能
public class BigStar implements Star { private String name; public BigStar() { } public BigStar(String name) { this.name = name; } //唱歌 @Override public String sing(String name){ System.out.println(this.name + "正在唱" + name); return "谢谢"; } //跳舞 @Override public void dance(){ System.out.println(this.name + "正在跳舞"); } /** * 获取 * @return name */ public String getName() { return name; } /** * 设置 * @param name */ public void setName(String name) { this.name = name; } public String toString() { return "BigStar{name = " + name + "}"; } }
代理接口 (指定被代理的所有方法)
就是要增加的功能的方法抽象出来
public interface Star { //中介公司 //我们可以把所有想要被代理的方法定义在接口当中 //唱歌 public String sing(String name); //跳舞 public void dance(); }
代理工具类,用于建立一个代理
测试类就可以调用下面的代码创建一个代理,运行指定鸡哥方法时,先跑代理的功能代码。
这样鸡哥只用唱跳了
public class ProxyUtil { /* * 方法的作用: * 给一个明星的对象,创建一个代理 * * 形参: * 被代理的明星对象 * * 返回值: * 给明星创建的代理 * * 需求: * 外面的人想要大明星唱一首歌 * 1. 获取代理的对象 * 代理对象 = ProxyUtil.createProxy(大明星的对象); * 2. 再调用代理的唱歌方法 * 代理对象.唱歌的方法("只因你太美"); * */ public static Star createProxy(BigStar bigStar){ /* java.lang.reflect.Proxy类:提供了为对象产生代理对象的方法: public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) 参数一:用于指定用哪个类加载器,去加载生成的代理类 参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法 参数三:用来指定生成的代理对象要干什么事情*/ Star star = (Star)Proxy.newProxyInstance( ProxyUtil.class.getClassLoader(), new Class[]{Star.class}, new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws InvocationTargetException, IllegalAccessException { if ("dance".equals(method.getName())){ System.out.println("代理对象在唱歌前,准备场地,收钱"); }else if ("sing".equals(method.getName())){ System.out.println("代理对象在唱歌前,准备话筒,收钱"); } return method.invoke(bigStar,args); } } ); return star; } }
测试类
public class Test { public static void main(String[] args) { BigStar bigStar = new BigStar("鸡哥"); Star star = ProxyUtil.createProxy(bigStar); String str = star.sing("只因太美"); System.out.println(str); star.dance(); } }
输出结果