Java Proxy

试想一个场景:一个节目想要邀请一个大牌明星去做一个演讲,这个节目组直接去沟通的一般不是这个明星本人,而是明星的经纪人,明星的时间,出场费,都尽在他的经纪人的掌握中,节目组只需要跟经纪人联系好一切事情就好,经纪人会告诉明星他跟节目组商量好的演出的时间地点等等。

这个的java实现很简单,只需要明星(Star)和经纪人(Handler)实现同一个接口(Human),并且把明星当做经纪人的一个程序变量引用进来就行:

public interface Human {

    public void speak();

}

public class Star implements Human {

    @Override

    public void speak() {

        System.out.println("LaLaLa");

    }

}

public class Handler implements Human {

    Human star = new Star();

    public Handler(Human human) {

        this.star = human;

    }

    @Override

    public void speak() {

        System.out.println("-----before-----");

        star.speak();

        System.out.println("-----after-----");

    }

}

测试类:

public class Main {

    public static void main(String[] args) {

        Human human = new Star();

        Handler handler = new Handler(human);

        handler.speak();

    }

}

上边的实现看似完美,其实不然,它主要有以下两个缺点:

1 代理类Handler跟目标类Star绑定死了,也就是说代理类只能代理Star这一个目标类,你如果有其他的目标类,你需要再写一个代理类去代理。映射到现实情况,每个明星都有一个代理人,但是一个代理人可能会同时代理多个明星,但是上边的实现显然不能实现一对多的情形。

2 代理类和目标类都需要实现同一个接口,如果接口增加方法的话将是一个灾难

显然我们需要一个更优秀的代理来让我们达到更好的效果,jdk的Proxy和InvocationHandler就是来做这个的:

Interface和Star的代码不变,Handler和Main的代码:

public class Handler implements InvocationHandler {

    Object target;

    public Handler(Object target) {

        this.target = target;

    }

    @Override

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

        System.out.println("-----before-----");

        method.invoke(target, args);

        System.out.println("-----after-----");

        return null;

    }

}

public class Main {

    public static void main(String[] args) {

        Human star = new Star();

        Handler handler = new Handler(star);

        Human proxy = (Human) Proxy.newProxyInstance(star.getClass().getClassLoader(), star.getClass().getInterfaces(), handler);

        proxy.speak();

    }

}

OK,这个实现足够好,它允许我们只用一个代理类就能代理多个目标类,代理类也不用实现接口,但是这种方式只能代理接口,不能直接代理实现类,这不够灵活,让我们来看看cglib吧:

Interface和Star的代码同样不变,Handler和Main的代码:

public class Handler implements MethodInterceptor {

    @Override

    public Object intercept(Object target, Method method, Object[] arg, MethodProxy methodProxy) throws Throwable {

        System.out.println("-----before-----");

        methodProxy.invokeSuper(target, arg);

        System.out.println("-----after-----");

        return null;

    }

}

public class Main {

    public static void main(String[] args) {

        Handler handler = new Handler();

        Enhancer enhancer = new Enhancer();

        enhancer.setSuperclass(Star.class);

        enhancer.setCallback(handler);

        Star o = (Star) enhancer.create();

        o.speak();

    }

}

从Main代码来看,Star不用实现接口也能被cglib所代理,这正是我们的最终追求目标。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 一、代理的概念与作用 1.1、生活中的代理 杭州人从在杭州本地从杭州的代理商(线下商店)中买联想电脑和直接跑到北京...
    侠客有情剑无情QAQ阅读 12,628评论 3 13
  • 1. 动态代理概念: 代理对象存在的价值: 主要用于拦截对真实业务对象的访问。 代理对象有什么方法: 一般来说,真...
    _凌浩雨阅读 9,241评论 0 1
  • 代理模式 ****代理模式的作用:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个客户不想或者不能...
    linheimx阅读 4,014评论 0 1
  • 在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。在代理模式...
    _凌浩雨阅读 3,701评论 0 1
  • 從小到大,都會聽到旁人教育要勤力要吃苦。 到人大了,自問比任何人都更能吃苦更勤力。 但慢慢發覺原來勤力是不足的, ...
    真_80后阅读 1,364评论 0 0

友情链接更多精彩内容