Java 代理模式有如下几种实现方式:

  1.静态代理。    

2.JDK动态代理。

3.CGLIB动态代理。

示例,有一个打招呼的接口。分别有两个实现,说hello,和握手。代码如下。

接口:

public interface Greeting {

    public void doGreet();
}

实现类:

public class SayHello implements Greeting {

    @Override
public void doGreet() {
System.out.println("Greeting by say 'hello' .");
} }
public class ShakeHands implements Greeting {

    @Override
public void doGreet() {
System.out.println("Greeting by shake others's hands .");
} }
public class KissHello  {

    public void doGreet() {
System.out.println("Greeting by kiss . ");
} }

在不改变代码的情况下,想在执行目标方法 前后 做一些其他操作。则可以通过代理方式来实现。

1.静态代理。需要创建代理类。代理类实现了和目标类一样的接口,代理类接收目标类对象,并在实现方法中调用目标类的实现方法前后做手脚。如下:

public class GreetStaticProxy implements Greeting {

    private Greeting hello;//被代理对象
public GreetStaticProxy(Greeting hello){
this.hello=hello;
} @Override
public void doGreet() {
before();//执行其他操作
this.hello.doGreet();//调用目标方法
after();//执行其他操作
} public void before(){
System.out.println("[StaticProxy] Come to someone.");
}
public void after(){
System.out.println("[StaticProxy] Back to his own corner");
}
}

测试调用:

public class Main {

    public static void main(String[] args) {
Greeting hello=new SayHello();
Greeting shakeHands=new ShakeHands(); //静态代理
GreetStaticProxy staticHelloProxy=new GreetStaticProxy(hello);
staticHelloProxy.doGreet();
System.out.println();
GreetStaticProxy shakeHandsProxy=new GreetStaticProxy(shakeHands);
shakeHandsProxy.doGreet();
}
运行结果:
[StaticProxy] Come to someone.
Greeting by say 'hello' .
[StaticProxy] Back to his own corner [StaticProxy] Come to someone.
Greeting by shake others's hands .
[StaticProxy] Back to his own corner

这个方式有弊端,如果有N个接口的实现类需要被代理,则需要创建N个代理类。

2.JDK动态代理

创建代理类,如下:

public class JdkProxy implements InvocationHandler {

    private Object target;

    public JdkProxy(Object obj){
this.target=obj;
} @Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable { Object result=null;
before();
result=method.invoke(target, args);
after();
return result;
} public void before(){
System.out.println("[JdkProxy] Come to someone.");
}
public void after(){
System.out.println("[JdkProxy] Back to his own corner");
}
}

测试调用:

public class Main {

    public static void main(String[] args) {
Greeting hello=new SayHello();
Greeting shakeHands=new ShakeHands(); //jdk动态代理
JdkProxy dynamicProxy=new JdkProxy(hello);
Greeting target=(Greeting) Proxy.newProxyInstance(hello.getClass().getClassLoader(),
hello.getClass().getInterfaces(), dynamicProxy);
target.doGreet();
System.out.println(); } }

这种方式和第一种方式相比,虽然不需要创建很多代理类,

但是,他依赖与“被代理的对象需要实现接口” 即:在上面给出的代码示例中,动态代理可以代理SayHello和ShakeHands,却不能代理KissHello。因为KissHello没有实现接口。

3.CGLIB动态代理。

创建代理类:

public class CglibProxy implements MethodInterceptor {

    public static CglibProxy proxy=new CglibProxy();
private CglibProxy(){} public static CglibProxy getInstance(){
return proxy;
} public <T> T getProxy(Class<T> cls){
return (T) Enhancer.create(cls, this);
} @Override
public Object intercept(Object obj, Method method, Object[] arg,
MethodProxy proxy) throws Throwable {
Object result=null;
try {
before();
result= proxy.invokeSuper(obj, arg);
after();
} catch (Exception e) {
e.printStackTrace();
}
return result;
} public void before(){
System.out.println("[cglib] Come to someone.");
}
public void after(){
System.out.println("[cglib] Back to his own corner.");
}
}

调用示例:

public class Main {

    public static void main(String[] args) {

        //cglib代理
Greeting targetProxy=CglibProxy.getInstance().getProxy(SayHello.class);
targetProxy.doGreet();
System.out.println(); CglibProxy.getInstance().getInstance().getProxy(KissHello.class).doGreet(); } }

综上,CGLIB动态代理最好,spring框架也用到了CGLIB包。

java 代理的三种实现方式的更多相关文章

  1. Java多线程的三种实现方式

    java多线程的三种实现方式 一.继承Thread类 二.实现Runnable接口 三.使用ExecutorService, Callable, Future 无论是通过继承Thread类还是实现Ru ...

  2. java的list几种实现方式的效率(ArrayList、LinkedList、Vector、Stack),以及 java时间戳的三种获取方式比较

    一.list简介 List列表类,顺序存储任何对象(顺序不变),可重复. List是继承于Collection的接口,不能实例化.实例化可以用: ArrayList(实现动态数组),查询快(随意访问或 ...

  3. Java定时器的三种实现方式

    一.普通thread /** * 普通thread * 这是最常见的,创建一个thread,然后让它在while循环里一直运行着, * 通过sleep方法来达到定时任务的效果.这样可以快速简单的实现, ...

  4. Java数组的三种打印方式

    1.数组的输出的三种方式 一维数组: 定义一个数组 int[] array = {1,2,3,4,5}; (1)传统的for循环方式 for(int i=0;i<array.length;i++ ...

  5. Java反射的三种实现方式

    Foo foo = new Foo(); 第一种:通过Object类的getClass方法 Class cla = foo.getClass(); 第二种:通过对象实例方法获取对象 Class cla ...

  6. java 多线程编程三种实现方式

    一种是继承Thread类,一种是实现Runable接口,还有一种是实现callable接口: 有博主说只有前面2种方式,我个人愚见是三种,主要详细介绍下callable的使用: 三种线程的我的个人理解 ...

  7. java集合的三种遍历方式

    import java.util.ArrayList;  import java.util.Collection;import java.util.Iterator;public class Home ...

  8. java线程的三种实现方式

    线程实现的三种种方式: 一个是继承Thread类,实现run()方法: 一个是实现Runnable接口,实现run()方法: 一个是实现Callable接口,实现call()方法:该方式和实现Runn ...

  9. Java规范的三种注释方式:

    1.单行注释 // //单行注释 2.多行注释 /* */ /* 多行 注释 */ 3.文档注释[java特有的] /** */ ◆注释内容可以被JDK提供的工具javadoc所解析,生成一套以网页文 ...

随机推荐

  1. IntelliJ IDEA Community Edition 14.1.4下使用 Apache-Subversion搭建代码管理环境

    当前我的idea 版本是14.1.4. 1,)SVN Server下载与安装(https://www.visualsvn.com/server/): 因为我开发机是x64的,所以我优先下载 x64的 ...

  2. C#文件与流(FileStream、StreamWriter 、StreamReader 、File、FileInfo、Directory、directoryInfo、Path、Encoding)

    (FileStream.StreamWriter .StreamReader .File.FileInfo.Directory.DirectoryInfo.Path.Encoding)     C#文 ...

  3. HttpClient_002_中文乱码、HttpClient中文乱码透析、总结

    中文乱码原理代码: String s = "中文"; byte[] bs2 = s.getBytes("utf-8");//将s拆成:utf-8个体,注:utf ...

  4. android opengl

    引用:http://weimingtom.iteye.com/blog/1616972 二维坐标系变换为原点在左上角(测试用) * GLES * JOGL * LWJGL * libgdx(使用g2d ...

  5. Egret官方案例学习笔记

    1.资源记载方式 (1)Egret引擎是2.0.5. (2)resource/resource.json文件是: { "resources": [ { "name&quo ...

  6. db2常用命令(1)

    DB2常用命令   1.启动实例(db2inst1):实例相当于informix中的服务 db2start 2.停止实例(db2inst1): db2stop 3.列出所有实例(db2inst1) d ...

  7. asp.net 捕获throw

    <script type="text/javascript"> function pageLoad() { Sys.WebForms.PageRequestManage ...

  8. Uva 11324 最大团

    题目链接:http://vjudge.net/contest/141990#problem/B 题意: 给一张有向图G,求一个结点集数最大的结点集,是的该结点集中任意两个结点 u 和 v,满足: 要么 ...

  9. AtCoder Beginner Contest 050 ABC题

    A - Addition and Subtraction Easy Time limit : 2sec / Memory limit : 256MB Score : 100 points Proble ...

  10. Exception&Error

    Java异常处理 1:什么是异常 异常(Exception)也叫异常.在Java编程语言中,异常就是程序在运行过程中由于硬件设备问题.软件设计错误.缺陷等导致的程序错误. 1.1:想打开的文件不存在 ...