乐趣区

关于java:面试官问你设计模式-你还在回答单例-和-工厂-代理模式了解一下

面试官问你设计模式 你还在答复单例 和 工厂?代理模式理解一下

走过路过不要错过,上一篇讲了反射,能够看我上一篇文章或微信上搜寻:木子的昼夜编程。

这一篇筹备写一下动静代理模式。

1. 理由

先给你一个理由,为什么学动静代理

第一他是一种设计模式,在你工作中如果能够联合它的应用,那代码相对丑陋。

第二面试官们喜爱问的面试题就包含设计模式,如果把握这种设计模式,那你下一份工作的薪水可能就会高一点点。

第三你可能看过很多源码包含但不限于 Mybatis、Spring,他们的作者在编写代码的时候就充分利用了动静代理的思维,你把握了这个技术之后,就能很好的了解那些写框架的大佬们是怎么设计的框架,从而失去能力的晋升。

2. 动起来

2.1 先说代理

代理就是一个类代表另一个类提供性能,你能够了解成你朋友圈的微商。

厂商生产鞋子 -> 微商从厂商那里零售鞋子在朋友圈进行售卖 -> 你购买鞋子

微商就是代理,代理的就是鞋子厂商,而你就是那个应用鞋子厂商生产的鞋子然而你须要与微商打交道。

网上常常会看到一个 UML 图,在下也献丑画一个。

2.2 代理的优缺点

长处:我的总结就是高扩大、智能化、职责清晰

毛病:老本加大(你购买的鞋子可能变贵,运输工夫可能变长)

3. 先静再动

聊动静代理之前先写一个动态代理的例子。

package test;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 * 一个生存在互联网底层,做着增删改查的码农, 不谙世事的造作
 * @create 2021-10-16 9:39
 *
 * 鞋厂标准
 */
public interface AShoeFactory {
    /**
     *
     * @param name 购鞋者名字
     * @param num 购买鞋子数量
     */
    void sellingShoes(String name, Integer num);
}
package test;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 * 一个生存在互联网底层,做着增删改查的码农, 不谙世事的造作
 * @create 2021-10-16 9:44
 *
 * 具体的鞋厂:鸿星尔克
 */
public class ErkeFactory implements AShoeFactory  {

    @Override
    public void sellingShoes(String name, Integer num) {System.out.println(name+"购买了"+ num+"双鞋,放烟花庆贺!!");
    }
}
package test;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 * 代理类:朋友圈某某某
 */
public class ProxyWechatMoments implements AShoeFactory{public ProxyWechatMoments(){}

    public ProxyWechatMoments(ErkeFactory erkeFactory){this.erkeFactory = erkeFactory;}
    ErkeFactory erkeFactory;
    @Override
    public void sellingShoes(String name, Integer num) {erkeFactory.sellingShoes(name, num);
    }
}

测试:

import test.ErkeFactory;
import test.ProxyWechatMoments;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 */
public class Test {public static void main(String[] args) {
        // 实在提供能力的类
        ErkeFactory erkeFactory = new ErkeFactory();
        // 代理类
        ProxyWechatMoments proxy =  new ProxyWechatMoments(erkeFactory);
        // 激动生产
        proxy.sellingShoes("最最最最最小李子", 10);
    }
}

测试输入后果:

最最最最最小李子 购买了 10 双鞋,放烟花庆贺!!

4. 动起来

写了动态代理,动态代理实现比较简单。

毛病呢:扩大比拟麻烦,如果上述例子中再加一个匹克鞋厂,咱们就须要批改代码,会在代理类中用 if else 辨别要购买哪个牌子的鞋、

4.1 动静代理之 JDK
package test;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 * 动静代理类
 */
public class ProxyDynamicHandler implements InvocationHandler {
    private  AShoeFactory shoeFactory;
    public ProxyDynamicHandler(){}
    public ProxyDynamicHandler(AShoeFactory shoeFactory) {this.shoeFactory = shoeFactory;}

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 卖鞋前后做的事件 是代理模式中常常用到的性能,作为加强和过滤应用
        System.out.println("动静代理的益处,卖鞋前做一些事件");
        String name = (String)args[0];
        // 如果购买的人是 追忆流年时光 那我就不卖给他
        if ("追忆流年时光".equals(name)) {System.out.println("不卖,谢谢!");
        } else {
            // 这里用到了反射 没把握的人能够看我上一篇文章 或关注公众号:木子的昼夜编程
            method.invoke(shoeFactory, args);
        }
        System.out.println("动静代理的益处,卖鞋后做一些事件");
        return null;
    }
}
import sun.applet.AppletClassLoader;
import test.AShoeFactory;
import test.ErkeFactory;
import test.ProxyDynamicHandler;

import java.lang.reflect.Proxy;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 */
public class TestDynamic {public static void main(String[] args) {AShoeFactory factory = (AShoeFactory) Proxy.newProxyInstance(ErkeFactory.class.getClassLoader(),// 类加载器
                ErkeFactory.class.getInterfaces(),// 代理的类的所有接口 Jdk 动静代理是基于接口的
                // 如果学过 Spring 的动静代理的话应该晓得,默认如果你的 Service 有接口 Spring 会应用 Jdk 动静代理
                // 如果你的 Service 没有接口 Spring 会应用 CGLIB
                // 这个常识 是我记忆中的 对不对不负责 哈哈
                new ProxyDynamicHandler(new ErkeFactory()));
        factory.sellingShoes("正义的帅哥", 10);
        System.out.println("------------------- 富丽丽的分割线 ------------------------");
        factory.sellingShoes("追忆流年时光", 10);
    }
}

输入:

动静代理的益处,卖鞋前做一些事件
正义的帅哥 购买了 10 双鞋,放烟花庆贺!!
动静代理的益处,卖鞋后做一些事件
——————- 富丽丽的分割线 ————————
动静代理的益处,卖鞋前做一些事件
不卖,谢谢!
动静代理的益处,卖鞋后做一些事件

4.2 CGLIB 动静代理

CGLIB 是一个很牛 X 的代码生成库,常被用在一些 AOP 框架里。

CGLIB 底层应用了 ASM 来操作字节码

他是对字节码进行了操作,说实话不是很平安,然而用起来很爽。

就是没有束缚的技术最疯狂。

CGLIB 实现动静代理的形式就是认干爹,利用字节码批改性能创立一个被代理类(匹克鞋厂)的子类,提供性能的时候还是应用了被代理类的性能,就跟花钱找干爹一个情理

注:这里的干爹是那种,70 年代 80 年代 很纯正的那种两家关系好 就认干爹那种 不是古代意思。

pom.xml

<dependencies>
    <dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.3.0</version>
    </dependency>
</dependencies>
package test;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 */
public class PeakFactory {public void sellingShoes(String name, Integer num) {System.out.println(name+"购买了"+ num+"双匹克鞋,放烟花庆贺!!");
    }
}

自定义 MethodHandler:

package test;

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 */
public class ProxyDynamicByCglib implements MethodInterceptor {

    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        // 卖鞋前后做的事件 是代理模式中常常用到的性能,作为加强和过滤应用
        System.out.println("动静代理的益处,卖鞋前做一些事件");
        String name = (String)args[0];
        // 如果购买的人是 追忆流年时光 那我就不卖给他
        if ("追忆流年时光".equals(name)) {System.out.println("不卖,谢谢!");
        } else {System.out.println("看一下代理对象的父类是个啥:"+obj.getClass().getSuperclass());
            // 调用业务类(父类)的办法
            proxy.invokeSuper(obj, args);

        }
        System.out.println("动静代理的益处,卖鞋后做一些事件");
        // 本次测试没有返回 无需返回
        return null;
    }
}

测试:

import net.sf.cglib.proxy.Enhancer;
import test.*;

/**
 * @author 发现更多精彩  关注公众号:木子的昼夜编程
 */
public class TestDynamicCGLIB {public static void main(String[] args) {Enhancer enhancer = new Enhancer();
        // 重点 设置理论提供办法的类为父类 Cglib 应用本人的批改字节码性能 生成一个这个类的子类对象
        enhancer.setSuperclass(PeakFactory.class);
        enhancer.setCallback(new ProxyDynamicByCglib());// 自定义的 Intrceptor
        PeakFactory factory = (PeakFactory) enhancer.create();

        factory.sellingShoes("正义的帅哥", 10);
        System.out.println("------------------- 富丽丽的分割线 ------------------------");
        factory.sellingShoes("追忆流年时光", 10);
    }
}

输入:

动静代理的益处,卖鞋前做一些事件
看一下代理对象的父类是个啥:class test.PeakFactory
正义的帅哥 购买了 10 双匹克鞋,放烟花庆贺!!
动静代理的益处,卖鞋后做一些事件
——————- 富丽丽的分割线 ————————
动静代理的益处,卖鞋前做一些事件
不卖,谢谢!
动静代理的益处,卖鞋后做一些事件

5. 唠唠

代理的话我对 JDK 动静代理和 CGLIB 动静代理的底层理解不是很好,所以就只开展说了一下简略的利用。

其实大部分时候看架构源码,很多中央会用到代理模式,依据我的教训,会简略应用的就可以看懂架构中的代码流程,不必摸索太深,当然不是不激励你去摸索,常识说如果有工夫的话再去摸索,没工夫就先学会怎么用。

本文由博客一文多发平台 OpenWrite 公布!

退出移动版