• 反射(3)—动态、静态代理


    • 1.动态代理是指客户通过代理类来调用其他对象(被代理类)的方法,并且是程序在运行时根据需要动态创建目标类的代理对象

      • 使用场合:调试、远程方法调用

      • 原理:使用一个代理将对象包装起来,然后用代理对象取代原始对象,任何对原始对象的调用都要通过代理,代理对象决定是否以及何时将方法调用转到原始对象上

      • 代码:
    //接口
    interface Subject{
        void action();
    }
    
    //被代理类
    class RealSubject implements Subject{
    
        @Override
        public void action() {
            System.out.println("我是被代理类,记得执行我...");
        }
    
    }
    
    //动态创建代理类,而不是直接实现Subject,想要动态创建代理类要使用到一个接口InvocationHandler
    class MyInvocationHandler implements InvocationHandler{
    
        Object obj;//实现了接口的被代理类的对象的声明
        //该方法作用:①给被代理类的对象实例化②返回一个代理类对象
        public Object Blind(Object obj){//该类对象调用该方法时,传进去参数为一个对象
            this.obj = obj;//实现了创建对象功能:父类的obj声明指向每某个子类对象
            //动态的返回一个代理类对象,要用到另一个类(Proxy)的静态方法(newProxyInstance)去创建
            //该方法第一个参数动态获取被代理类类加载器,第二个参数动态是被代理类实现什么接口,
            //第三个是实现了InvocationHandler接口的类的对象
            return Proxy.newProxyInstance(obj.getClass().
                    getClassLoader(), obj.getClass().getInterfaces(), this);
        }
        //当代理类的对象发起对被重写的方法调用时,都会转换成调用如下方法
        @Override
        public Object invoke(Object proxy, Method method, Object[] args)
                throws Throwable {
            Object retrunValue = method.invoke(obj, args);
            return retrunValue;
        }
    
    }
    
    public class 动态代理 {
    
    
        public static void main(String[] args) {
            //1.创建被代理类对象 
            RealSubject rs = new RealSubject();
            //2.创建实现了InvocationHandler接口的类的对象,也就是MyInvocationHandler
            MyInvocationHandler mih = new MyInvocationHandler();
            //3.调用blind()方法,动态的返回一个同样的实现了-对象rs所在类 -实现的接口的代理类的对象
            //或则说动态返回一个对象,这个对象是和被代理类实现相同接口的代理类创建的
            Object obj = mih.Blind(rs);
            Subject sub = (Subject)obj;//此时sub就是代理类的对象
            sub.action();//调用action()方法,会自动转到实现InvocationHandler接口的实现 类中invoke方法的调用
    
            //再写一个例子,被代理类对象
            NikeClothFactory ncf = new NikeClothFactory();
            ClothFactory cf = (ClothFactory)mih.Blind(ncf);
            cf.protectCloth();
    
        }
    
    }
    
    • 2.静态代理模式:代理和被代理是成对出现的,有点麻烦
      • 功能:生产衣服
    //接口
    interface ClothFactory{
        void protectCloth();
    }
    
    //被代理类
    class NikeClothFactory implements ClothFactory{
        @Override
        public void protectCloth() {
            System.out.println("Nike工厂生产一批衣服");
        }
    }
    //代理类
    class ProxyClothFactory implements ClothFactory{
        ClothFactory cf;
        //创建代理类对象时,实际传入一个被代理类的对象
        public ProxyClothFactory(ClothFactory cf){
            this.cf = cf;
        }
        @Override
        public void protectCloth() {
            System.out.println("代理费100,代理类开始执行...");
            cf.protectCloth();
        }
    
    }
    
    
    public class 静态代理 {
    
        public static void main(String[] args) {
            NikeClothFactory nc = new NikeClothFactory();//创建被代理类对象
            ProxyClothFactory pc = new ProxyClothFactory(nc);//创建代理类对象
    
            pc.protectCloth();//此时调用的方法为被代理类中的方法
    
    
        }
    
    }
    
  • 相关阅读:
    用批处理来启动/停止SQL SERVER 2005的服务 【转载】
    c#命名法 【转】
    oracle 隐式游标,显示游标,游标循环,动态SELECT语句和动态游标,异常处理,自定义异常【转载】
    fetch bulk collect into 批量效率的读取游标数据 【转载】
    Oracle 外连接和 (+)号的用法 【转载】
    如何在Oracle中复制表结构和表数据 【转载】
    Oracle 小知识点
    VSS 2005 配置(含录像) 【转载】
    json 详解 【转】
    .NET 2.0 使用最新版的JSON.net 进行反序列化 【转载】
  • 原文地址:https://www.cnblogs.com/tengpengfei/p/10453996.html
Copyright © 2020-2023  润新知