14.动态代理

代理模式

java动态代理:利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。

cglib动态代理:利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

  1. 概念
    1. 真实对象:被代理的对象
    2. 代理对象
    3. 代理模式:(代理对象代理真实对象,达到增强真实对象功能的目的)
  2. 实现方式:
    1. 静态代理:有一个类文件描述代理模式
    2. 动态代理:在内存中形成代理类
      1. 实现步骤:
        1. 代理对象和真实对象实现相同的接口
        2. 代理对象 = Proxy.newProxyInstance();
        3. 使用代理对象调用方法。
        4. 增强方法
      2. 增强方式:
        1. 增强参数列表
        2. 增强返回值类型
        3. 增强方法体执行逻辑

代理对象和被代理对象实现了相同的一些接口必须是相同的!由于是面向接口编程,多态的思想,所以可以不改变源码的基础上添加功能,可扩展性高,非常方便。


动态代理终极版.png
  1. java.lang.reflex.Proxy.newProxyInstance方法(下文简称“NPI方法”)创建代理对象的步骤

NPI方法需要三个参数,
①加载相关接口的类加载器loader、
②代理对象要实现的接口类对象interface数组、
③代理对象最终执行方法所属的匿名内部类IH并实现该接口的invoke方法
NPI方法利用反射机制通过类加载器将代理对象要实现的接口类加载到方法区,在堆内存中创建了代理对象,这个代理对象可牛逼了,实现了接口中的各个方法(JDK的底层实现细节,暂时不懂!而且似乎是可变长列表),然后在各个方法中通过反射机制获得了对应方法的类对象Method,然后在方法内调用匿名内部类IH的invoke方法,传入三个参数,
①代理对象本身的引用this、
②通过反射机制实现的 调用匿名内部类IH的invoke方法的方法所对应的Method对象、
③方法对象运行需要的对应参数。
最后,NPI方法返回代理对象的引用

  1. 代理对象调用代理方法执行流程如下

调用 代理对象.代理方法(代理参数可变长列表)
会找到对应的代理方法,传入代理参数并入栈执行。
这个代理方法中会生成被代理方法对应的Method对象,然后执行匿名内部类的invoke方法,匿名内部类invoke方法中传入的参数为代理对象、代理方法,传入的代理参数列表;在匿名内部类invoke方法中就可以实现 ①增强传入的参数②增强返回值③在被代理对象的方法执行前后添加逻辑代码; 代理对象通过Method对象执行被代理对象的方法(反射机制)method.invoke(被代理的对象,参数)其实,说白了,动态代理的本质就是执行匿名内部类的invoke方法,这个方法内可以在被代理方法执行前后以及被代理方法的参数、返回值进行一些装饰加工。python中的装饰器也可以实现这种功能,python中可以直接将函数当做参数,而且可以在函数内嵌套定义函数,理解起来很容易!

如下案例演示:

  1. 被代理对象实现的接口
public interface SaleComputer {
    public String sale(double money);
    public void show();
    public Object test();
}
  1. 被代理对象的逻辑代码
public class Lenovo implements SaleComputer {
    @Override
    public String sale(double money) {
        System.out.println("花了" + money + "元买了一台联想电脑");
        return "联想电脑"; }
    // 展示电脑
    public void show() {
        System.out.println("展示电脑...."); }
    @Override
    public Object test() { return null; }
}
  1. 使用代理增强案例
public static void main(String[] args)
...
// 1.创建真实对象
Lenovo lenovo = new Lenovo();

/** 和【python】中的【函数装饰器@】功能如出一辙
 * 三个参数:
 *      1. 类加载器:真实对象.getClass().getClassLoader()
 *      2. 接口数组:真实对象.getClass()..getInterfaces()
 *      3. 处理器:new InvocationHandler() 匿名内部内实现
 *          invoke(代理对象,代理方法,代理方法参数) 方法
 */

// 使用动态代理增强lenovo对象
SaleComputer proxy_lenovo = (SaleComputer)Proxy.newProxyInstance(
    lenovo.getClass().getClassLoader(),  // 负责加载相关的接口
    lenovo.getClass().getInterfaces(),   // 被加载执行的接口
    new InvocationHandler() {  // 接口协议,
        /**
         * 代理逻辑编写的方法,代理对象调用的所有方法都会触发该方法执行。
         * 参数:
         *      1. proxy: 代理对象
         *      2. method: 代理对象调用的方法,被封装为的对象。
         *      3. args: 代理对象调用方法时,传递的实际参数。
         */
        @Override
        public Object invoke(Object proxy,  // 代理对象,就是proxy_lenovo,方法里面使用this访问的为该匿名内部类
                             Method method, // 被代理执行的方法
                             Object[] args) // 被代理方法的参数
                throws Throwable {

            if(method.getName().equals("sale")) {
                // 1.增强参数案例演示【修改函数执行的参数】
                double money = (double)args[0] * 0.85;

                // 2.1 增强方法逻辑演示【在方法执行前后加一些代码】
                System.out.println("==========");
                System.out.println("专车接.....");

                // @@@@@@@ 使用真实对象调用该方法 @@@@@@@
                Object obj = method.invoke(lenovo, money);
                // @@@@@@@ ================== @@@@@@@

                // 2.2 增强方法逻辑演示
                System.out.println("专车送.....");
                System.out.println("==========");

                // 3. 增强返回值演示【在返回值上做手脚】
                return (String)obj + ",赠送一个_美女机器人";
            } else if(method.getName().equals("test"))
                 return proxy;
             else
                 return method.invoke(lenovo, args);
        }
    }
);

// sale方法被代理增强了
String computer = proxy_lenovo.sale(8000);
System.out.println(computer);

// show方法原样调用
proxy_lenovo.show();

// test方法进行测试
Object obj = proxy_lenovo.test();
System.out.println("判断是为同一个对象!:" + (obj == proxy_lenovo)); // true
}

filter拦截器过滤敏感词汇案例

思路:返回值增强,在调用被代理方法之后对参数进行判断替换!

  • 步骤,
    1. 实现动态代理对象
    2. 将动态代理对象放行
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.HashMap;
import java.util.Map;

@WebFilter("/*")
public class SensitiveFilter implements Filter {
    private String[] sensitive = null;
    /**
     * 敏感词汇过滤器,使用动态代理增强的方式
     * 进行加强,使用返回值加强方式进行加强
     */
    public void doFilter(ServletRequest req, ServletResponse resp, FilterChain chain) throws ServletException, IOException {

        ServletRequest newProxyInstance = (ServletRequest)Proxy.newProxyInstance(
            req.getClass().getClassLoader(),
            req.getClass().getInterfaces(),
            new InvocationHandler() {
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

                    // 判断getParameter方法!
                    if (method.getName().equals("getParameter")) {
                        String str = (String)method.invoke(req, args);
                        for (String s : sensitive) {
                            if (str.contains(s))
                                str = str.replaceAll(s, "***");
                        }
                        return str;
                    }

                    // 判断getParameterMap方法
                    // Map<String, String[]> map = req.getParameterMap();
                    if(method.getName().equals("getParameterMap")) {
                        Map<String, String[]> map = (Map<String, String[]>)method.invoke(req, args);
                        Map<String, String[]> newMap = new HashMap<>();
                        for (String s : map.keySet()) {
                            String[] temp = map.get(s);
                            for (int i = 0; i < temp.length; i++) {
                                for (String m : sensitive) {
                                    if(temp[i].contains(m))
                                        temp[i] = temp[i].replaceAll(m, "***");
                                }
                            }
                            newMap.put(s, temp);
                        }
                        return newMap;
                    }

                    return method.invoke(req, args);
                }
            }
        );

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

推荐阅读更多精彩内容