美文网首页
java 代理

java 代理

作者: 子鱼城 | 来源:发表于2017-10-04 21:52 被阅读22次

    java的代理有两种: 静态代理和动态代理,在一些比较高级的功能中就会使用到java代理,这里复习一下java的代理。

    代理的功能:可以做到: 修改function的参数和返回值function执行前后添加部分功能

    代理模式是常用的java设计模式,特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。

    静态代理

    静态代理就是实现一个类Proxy, 类Proxy实现了类A的interface ICheckListener, 并在实现的代码里面调用类A的接口,对外提供Proxy, 不提供A,具体看代码。

    interface ICheckListener() {
        void onCheck(int num);
    }
    
    class CheckImpl implements ICheckListener {
        @Overrude
        public void onCheck(int num) {
            if (num > 10) {
                Logger.i(TAG, "bigger than 10")
           } else {
                Logger.i(TAG, "smaller than 10");
           }
        }
    }
    
    Class CheckImplProxy implments ICheckListener {
         CheckImpl a;
         public CheckImplProxy(CheckImpl a) {
             this.a = a; 
        }
        @Override
        public void onCheck(int num) {
             num = num - 10;
             Logger(TAG, "begin to change");
             if (this.a != null) {
                  a.onCheck(num);
             }
             Logger(TAG, "end to change");
        }
    }
    
    

    在使用到CheckImpl的实例的地方把CheckImpl替换成CheckImplProxy的实例,对外是无感知的行为,当调用onCheck()方法时,CheckImplProxy的实例就会将参数减少10 并且在调用开始和结束的地方都打上log。

    实现静态代理,最主要的是找到想要替换的实例(通常是静态实例或者单例类),生成 hook后的实例,取代要替换的实例。

    静态代理的缺点:

    1. 由于代理类需要实现所有的原始类的方法,当接口出现变动时, 代理类同样需要跟着修改,这样就出现代码冗余以及代码维护不方便的问题
    2. 一个代理类只能服务于一种类型的对象(例如CheckImpl),当需要为另外的类(例如ShoppingImpl)实现代理类时,我们不得不重新编写新的代理类.

    动态代理

    所以我们就会想办法可以通过一个代理类完成全部的代理功能,那么我们就需要用动态代理
    在Java中要想实现动态代理机制,需要java.lang.reflect.InvocationHandler接口和 java.lang.reflect.Proxy 类的支持

    //interface
    public interface IShopAction {
        void doShop(int count);
    }
    
    
    //impl
    public class DynamicProxyExample implements IShopAction {
        public static String TAG = DynamicProxyExample.class.getSimpleName();
        @Override
        public void doShop(int count) {
            Log.d(TAG, "buy " +  count + " item in shop");
        }
    }
    
    
    
    
    //invocationHandler
    public class ShopInvocationHandler implements InvocationHandler {
        Object target ;
        public ShopInvocationHandler(Object example) {
            target = example;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            Log.d("cheng", "method name:"+method.getName() + " class name:" + proxy.getClass().getSimpleName() + " args length:"+ args.length);
            if ("doShop".equals(method.getName()) && proxy instanceof IShopAction) {
                Log.d("cheng", "proxy begin work");
    
                //如果是dynamicproxyexample 的doshop方法
                int num = (int)args[0];
                num = num / 2;
                //修改参数
                Log.d("cheng", num +"购买的商品数量");
                //执行购买操作
                Object result =  method.invoke(target, num);
                //操作完
                Log.d("cheng", "proxy end work");
                return result;
            }
    
            return null;
        }
    }
    
    
    //test
    public class DynamicProxyExampleTest {
        public static void doShopActionTest() {
            DynamicProxyExample example = new DynamicProxyExample();
            InvocationHandler handler = new ShopInvocationHandler(example);
            IShopAction exampleProxy = (IShopAction) Proxy.newProxyInstance(example.getClass().getClassLoader(), example.getClass().getInterfaces(), handler);//这里转换类型只能强制转换为接口类型,不能转换为其他类型。
            exampleProxy.doShop(10);
    
            DynamicProxyExample example = new DynamicProxyExample();
            example.doShop(10);
        }
    }
    

    实现InvocationHandler的invoke方法,判断方法名以及接口名,如果是我们希望处理的方法以及接口,则获取方法名,参数名,在调用实际方法前,打印log日志,调用实际方法后,同样打印一条log日志,运行程序后,发现i的确走到了invocationHandler里面了。

    以上就是jdk的动态代理, jdk的动态代理要求被代理的类实现了某个接口,如果需要被代理的类没有实现接口,则不能用这种方式实现动态代理, 可以使用cglib的动态代理。

    refs:

    相关文章

      网友评论

          本文标题:java 代理

          本文链接:https://www.haomeiwen.com/subject/ftmcyxtx.html