代理

《转》JAVA动态代理(JDK和CGLIB)

代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

该文章转自:

http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html

JAVA的动态代理

代理模式

代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

按照代理的创建时期,代理类可以分为两种。

静态代理

由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。

动态代理

在程序运行时,运用反射机制动态创建而成。

首先看一下静态代理:

1、Count.java

/**

* 定义一个账户接口

*

* @author Administrator

*

*/publicinterfaceCount{// 查看账户方法publicvoidqueryCount();// 修改账户方法publicvoidupdateCount();  }

2、CountImpl.java

/**  * 委托类(包含业务逻辑)  *  *@authorAdministrator  *  */publicclassCountImplimplementsCount{@OverridepublicvoidqueryCount(){          System.out.println("查看账户方法...");      }@OverridepublicvoidupdateCount(){          System.out.println("修改账户方法...");      }  }  、CountProxy.javapackagenet.battier.dao.impl;importnet.battier.dao.Count;/**  * 这是一个代理类(增强CountImpl实现类)  *  *@authorAdministrator  *  */publicclassCountProxyimplementsCount{privateCountImpl countImpl;/**      * 覆盖默认构造器      *      *@paramcountImpl      */publicCountProxy(CountImpl countImpl){this.countImpl = countImpl;      }@OverridepublicvoidqueryCount(){          System.out.println("事务处理之前");// 调用委托类的方法;countImpl.queryCount();          System.out.println("事务处理之后");      }@OverridepublicvoidupdateCount(){          System.out.println("事务处理之前");// 调用委托类的方法;countImpl.updateCount();          System.out.println("事务处理之后");      }  }

3、TestCount.java

packagenet.battier.test;importnet.battier.dao.impl.CountImpl;importnet.battier.dao.impl.CountProxy;/**  *测试Count类  *  *@authorAdministrator  *  */publicclassTestCount{publicstaticvoidmain(String[] args){          CountImpl countImpl =newCountImpl();          CountProxy countProxy =newCountProxy(countImpl);          countProxy.updateCount();          countProxy.queryCount();      }  }

观察代码可以发现每一个代理类只能为一个接口服务,这样一来程序开发中必然会产生过多的代理,而且,所有的代理操作除了调用的方法不一样之外,其他的操作都一样,则此时肯定是重复代码。解决这一问题最好的做法是可以通过一个代理类完成全部的代理功能,那么此时就必须使用动态代理完成。

再来看一下动态代理:

JDK动态代理中包含一个类和一个接口:

InvocationHandler接口:

public interface InvocationHandler {

public Object invoke(Object proxy,Method method,Object[] args) throws Throwable;

}

参数说明:

Object proxy:指被代理的对象。

Method method:要调用的方法

Object[] args:方法调用时所需要的参数

可以将InvocationHandler接口的子类想象成一个代理的最终操作类,替换掉ProxySubject。

Proxy类:

Proxy类是专门完成代理的操作类,可以通过此类为一个或多个接口动态地生成实现类,此类提供了如下的操作方法:

public static Object newProxyInstance

(ClassLoader loader, Class[] interfaces, InvocationHandler h) throws IllegalArgumentException

参数说明:

ClassLoader loader:类加载器

Class[] interfaces:得到全部的接口

InvocationHandler h:得到InvocationHandler接口的子类实例

Ps:类加载器

在Proxy类中的newProxyInstance()方法中需要一个ClassLoader类的实例,ClassLoader实际上对应的是类加载器

在Java中主要有一下三种类加载器;

Booststrap ClassLoader:此加载器采用C++编写,一般开发中是看不到的;

Extendsion ClassLoader:用来进行扩展类的加载,一般对应的是jre\lib\ext目录中的类;

AppClassLoader:(默认)加载classpath指定的类,是最常使用的是一种加载器。

动态代理

与静态代理类对照的是动态代理类,动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java反射机制可以生成任意类型的动态代理类。java.lang.reflect包中的Proxy类和InvocationHandler接口提供了生成动态代理类的能力。

动态代理示例:

1、BookFacade.java

packagenet.battier.dao;publicinterfaceBookFacade{publicvoidaddBook();  }

2、BookFacadeImpl.java

packagenet.battier.dao.impl;importnet.battier.dao.BookFacade;publicclassBookFacadeImplimplementsBookFacade{@OverridepublicvoidaddBook(){          System.out.println("增加图书方法。。。");      }  }

、BookFacadeProxy.java

packagenet.battier.proxy;importjava.lang.reflect.InvocationHandler;importjava.lang.reflect.Method;importjava.lang.reflect.Proxy;/**  * JDK动态代理代理类  *  *@authorstudent  *  */publicclassBookFacadeProxyimplementsInvocationHandler{privateObject target;/**      * 绑定委托对象并返回一个代理类      *@paramtarget      *@return*/publicObjectbind(Object target){this.target = target;//取得代理对象returnProxy.newProxyInstance(target.getClass().getClassLoader(),                  target.getClass().getInterfaces(),this);//要绑定接口(这是一个缺陷,cglib弥补了这一缺陷)}@Override/**

* 调用方法

*/publicObjectinvoke(Object proxy, Method method, Object[] args)throwsThrowable{          Object result=null;          System.out.println("事物开始");//执行方法result=method.invoke(target, args);          System.out.println("事物结束");returnresult;      }  }

3、TestProxy.java

packagenet.battier.test;importnet.battier.dao.BookFacade;importnet.battier.dao.impl.BookFacadeImpl;importnet.battier.proxy.BookFacadeProxy;publicclassTestProxy{publicstaticvoidmain(String[] args){          BookFacadeProxy proxy =newBookFacadeProxy();          BookFacade bookProxy = (BookFacade) proxy.bind(newBookFacadeImpl());          bookProxy.addBook();      }  }

但是,JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK代理,这就要使用cglib动态代理了。

Cglib动态代理

JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。

示例

1、BookFacadeCglib.java

publicinterfaceBookFacade{publicvoidaddBook();  }

2、BookCadeImpl1.java

package net.battier.dao.impl;/**

* 这个是没有实现接口的实现类

*

* @author student

*

*/publicclassBookFacadeImpl1{publicvoidaddBook(){          System.out.println("增加图书的普通方法...");      }  }

3、BookFacadeProxy.java

packagenet.battier.proxy;importjava.lang.reflect.Method;importnet.sf.cglib.proxy.Enhancer;importnet.sf.cglib.proxy.MethodInterceptor;importnet.sf.cglib.proxy.MethodProxy;/**  * 使用cglib动态代理  *  *@authorstudent  *  */publicclassBookFacadeCglibimplementsMethodInterceptor{privateObject target;/**      * 创建代理对象      *      *@paramtarget      *@return*/publicObjectgetInstance(Object target){this.target = target;          Enhancer enhancer =newEnhancer();          enhancer.setSuperclass(this.target.getClass());// 回调方法enhancer.setCallback(this);// 创建代理对象returnenhancer.create();      }@Override// 回调方法publicObjectintercept(Object obj, Method method, Object[] args,

MethodProxy proxy)throwsThrowable{          System.out.println("事物开始");          proxy.invokeSuper(obj, args);          System.out.println("事物结束");returnnull;      }  }

4、TestCglib.java

packagenet.battier.test;importnet.battier.dao.impl.BookFacadeImpl1;importnet.battier.proxy.BookFacadeCglib;publicclassTestCglib{publicstaticvoidmain(String[] args){          BookFacadeCglib cglib=newBookFacadeCglib();          BookFacadeImpl1 bookCglib=(BookFacadeImpl1)cglib.getInstance(newBookFacadeImpl1());          bookCglib.addBook();      }  }

以上为转载内容,下面是自己写的Test

packagecom.xt.test;importjava.lang.reflect.InvocationHandler;importjava.lang.reflect.Method;importjava.lang.reflect.Proxy;interfaceAnim{publicvoidsound();publicvoidname();}classDogimplementsAnim{@Overridepublicvoidsound(){// TODO Auto-generated method stubSystem.out.println("汪汪汪!");    }@Overridepublicvoidname(){// TODO Auto-generated method stubSystem.out.println("一条小狗!");    }}classCatimplementsAnim{@Overridepublicvoidsound(){// TODO Auto-generated method stubSystem.out.println("喵喵喵!");    }@Overridepublicvoidname(){// TODO Auto-generated method stubSystem.out.println("一只小猫!");    }}interfaceCar{publicvoidname();publicvoidcolor();publicvoidrun();}classBmwimplementsCar{@Overridepublicvoidname(){// TODO Auto-generated method stubSystem.out.println("一辆宝马!");    }@Overridepublicvoidcolor(){// TODO Auto-generated method stubSystem.out.println("黑色!");    }@Overridepublicvoidrun(){// TODO Auto-generated method stubSystem.out.println("车子开动!");    }}classWmimplementsCar{@Overridepublicvoidname(){// TODO Auto-generated method stubSystem.out.println("一辆大众!");    }@Overridepublicvoidcolor(){// TODO Auto-generated method stubSystem.out.println("银色!");    }@Overridepublicvoidrun(){// TODO Auto-generated method stubSystem.out.println("车子开动!");    }}/** * JDK动态代理代理类 *  *@authorstudent *  */classDynamicProxyimplementsInvocationHandler{privateObject target;/**    * 绑定委托对象并返回一个代理类    *      *@paramtarget    *@return*/publicObjectbind(Object target){this.target = target;// 取得代理对象Object o = Proxy.newProxyInstance(target.getClass().getClassLoader(),                target.getClass().getInterfaces(),this);// 要绑定接口(这是一个缺陷,cglib弥补了这一缺陷)returno;    }@Override/**

* 调用方法

*/publicObjectinvoke(Object proxy, Method method, Object[] args)throwsThrowable{        Object result =null;if(targetinstanceofAnim && method.getName().equals("sound"))            System.out.println("准备叫:");if(targetinstanceofCar && method.getName().equals("run"))            System.out.println("车子点火!");// 执行方法result = method.invoke(target, args);if(targetinstanceofAnim && method.getName().equals("sound"))            System.out.println("叫响了");if(targetinstanceofCar && method.getName().equals("run"))            System.out.println("车子停了!");returnresult;    }}publicclassDynamicProxyTest{publicstaticvoidmain(String[] args){        DynamicProxy proxy =newDynamicProxy();        Anim dog = (Anim) proxy.bind(newDog());        dog.name();        dog.sound();        System.out.println("---------------------");        Anim cat = (Anim) proxy.bind(newCat());        cat.name();        cat.sound();        System.out.println("---------动态的精髓来了,无需再构建代理,不同接口使用相同代理即可------------");// 动态的精髓来了,无需再构建代理,不同接口使用相同代理即可Car bmw = (Car) proxy.bind(newBmw());        bmw.name();        bmw.color();        bmw.run();        System.out.println("---------------------");        Car wm = (Car) proxy.bind(newWm());        wm.name();        wm.color();    }}输出结果:一条小狗!准备叫:汪汪汪!叫响了---------------------一只小猫!准备叫:喵喵喵!叫响了---------动态的精髓来了,无需再构建代理,不同接口使用相同代理即可------------一辆宝马!黑色!车子点火!车子开动!车子停了!---------------------一辆大众!银色!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容