Spring设计模式——代理模式[手写实现JDK动态代理]
代理模式
代理模式(Proxy Pattern):是指为其他对象提供一种代理,以控制对这个对象的访问。
代理对象在客户端和目标对象之间起到中介作用,代理模式属于结构型设计模式。
使用代理模式主要有两个目的:
- 一是保护目标对象
 - 二是增强目标对象
 
静态代理
package org.example.spring.designpattern.proxy.staticproxy;
/**
 * @author ss_419
 */
public interface Person {
    /**
     * 人有很多行为,要谈恋爱,要住房子,要购物,要工作
     */
    public void findLove();
}
package org.example.spring.designpattern.proxy.staticproxy;
/**
 * TODO 儿子要找对象,实现Son类
 *
 * @author ss_419
 * @version 1.0
 * @date 2023/3/9 10:46
 */
public class Son implements Person{
    /**
     *
     */
    @Override
    public void findLove() {
        // 我没有时间,我要赚钱......
        System.out.println("儿子要求:肤白貌美大长腿.....");
    }
}
package org.example.spring.designpattern.proxy.staticproxy;
/**
 * TODO 父亲要帮儿子相亲,实现Father类
 *
 * @author ss_419
 * @version 1.0
 * @date 2023/3/9 10:54
 */
public class Father {
    private Son son;
    // 没办法扩展
    public Father(Son son) {
        this.son = son;
    }
    // 获取目标对象的引用
    public void findLove(){
        System.out.println("父亲物色对象");
        this.son.findLove();
        System.out.println("双方同意交往,确立关系...");
    }
    public static void main(String[] args) {
        // 只能帮儿子找对象,不能帮表妹、不能帮陌生人
        Father father = new Father(new Son());
        father.findLove();
    }
}
动态代理
动态代理和静态代理的基本思路是一致的,只不过动态代理功能更加强大,随着业务的扩展适应性更强。
JDK实现方式:
package org.example.spring.designpattern.proxy.dynamicproxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * TODO 创建媒婆(婚介所)
 *
 * @author ss_419
 * @version 1.0
 * @date 2023/3/9 11:07
 */
public class JDKMeipo implements InvocationHandler {
    // 被代理的对象,将引用保存下来
    private Object target;
    public Object getInstance(Object target) throws Exception {
        this.target = target;
        System.out.println("JDKMeipo is instance of "+target.getClass().getName());
        Class<?> clazz = target.getClass();
        return Proxy.newProxyInstance(clazz.getClassLoader(), clazz.getInterfaces(), this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        before();
        Object invoke = method.invoke(this.target, args);
        after();
        return invoke;
    }
    private void before(){
        System.out.println("JDKMeipo before");
        System.out.println("我是媒婆:我要给你找对象,现在已经确认你的需求");
        System.out.println("..........开始物色..........");
    }
    private void after() {
        System.out.println("如果合适的话,就准备办事?");
    }
}
创建单身客户类Customer:
package org.example.spring.designpattern.proxy.dynamicproxy;
import org.example.spring.designpattern.proxy.staticproxy.Person;
/**
 * TODO
 *
 * @author ss_419
 * @version 1.0
 * @date 2023/3/9 11:57
 */
public class Customer implements Person {
    @Override
    public void findLove() {
        System.out.println("找到了一个人的心");
        System.out.println("高富帅");
        System.out.println("18块腹肌");
    }
}
创建测试:
package org.example.spring.designpattern.proxy.dynamicproxy;
import org.example.spring.designpattern.proxy.staticproxy.Person;
/**
 * TODO
 *
 * @author ss_419
 * @version 1.0
 * @date 2023/3/9 11:59
 */
public class DynamicProxyTest {
    public static void main(String[] args) {
        try {
            Person obj = (Person) new JDKMeipo().getInstance(new Customer());
            obj.findLove();
        }catch (Exception e) {
            e.printStackTrace();
        }
    }
}

手写实现JDK动态代理
不仅知其然,还得知其所以然。接下来我们自己动手写一个属于自己的动态代理
我们都知道JDK动态代理采用字节重组,重新生成对象来替代原始对象,以达到动态代理的目的。
JDK动态代理生成对象的步骤如下:
- 获取被代理对象的引用,并且获取它的所有接口,反射获取
 - JDK动态代理类重新生成一个新的类,同时新的类要实现被代理实现的所有接口。
 - 动态生成Java代码,新加的业务逻辑方法由一定的逻辑代码调用(在代码中体现)。
 - 编译新生成的Java代码.class文件
 - 重新加载到JVM中运行
 
以上的过程就叫字节码重组
JDK中有一个规范,在ClassPath下只要是$开头的.class文件,一般都是自动生成的。
第一步,创建Handler
package org.example.spring.designpattern.proxy.myproxy;
import java.lang.reflect.Method;
/**
 * TODO 1、创建接口
 * @author ss_419
 */
public interface GPInvocationHandler {
    /**
     * Invokes the method
     * @return
     */
    public Object invoke(Object proxy, Method method,Object[] args) throws Throwable;
}
第二步,用来生成源代码的工具类
package org.example.spring.designpattern.proxy.myproxy;
import javax.tools.JavaCompiler;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import java.io.File;
import java.io.FileWriter;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
/**
 * TODO 2、用来生成源代码的工具类
 *
 * @author ss_419
 * @version 1.0
 * @date 2023/3/9 13:48
 */
public class GPProxy {
    public static final String ln = "\r\n";
    public static Object newProxyInstance(GPClassLoader classLoader, Class<?>[] interfaces, GPInvocationHandler h) {
        try {
            //1、动态生成源代码.java文件
            String src = generateSrc(interfaces);
//           System.out.println(src);
            //2、Java文件输出磁盘
            String filePath = GPProxy.class.getResource("").getPath();
//           System.out.println(filePath);
            File f = new File(filePath + "$Proxy0.java");
            FileWriter fw = new FileWriter(f);
            fw.write(src);
            fw.flush();
            fw.close();
            //3、把生成的.java文件编译成.class文件
            JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
            StandardJavaFileManager manage = compiler.getStandardFileManager(null, null, null);
            Iterable iterable = manage.getJavaFileObjects(f);
            JavaCompiler.CompilationTask task = compiler.getTask(null, manage, null, null, null, iterable);
            task.call();
            manage.close();
            //4、编译生成的.class文件加载到JVM中来
            Class proxyClass = classLoader.findClass("$Proxy0");
            Constructor c = proxyClass.getConstructor(GPInvocationHandler.class);
            f.delete();
            //5、返回字节码重组以后的新的代理对象
            return c.newInstance(h);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
    private static String generateSrc(Class<?>[] interfaces) {
        StringBuffer sb = new StringBuffer();
        sb.append("package org.example.spring.designpattern.proxy.myproxy;" + ln);
        sb.append("import org.example.spring.designpattern.proxy.staticproxy.Person;" + ln);
        sb.append("import java.lang.reflect.*;" + ln);
        sb.append("public class $Proxy0 implements " + interfaces[0].getName() + "{" + ln);
        sb.append("GPInvocationHandler h;" + ln);
        sb.append("public $Proxy0(GPInvocationHandler h) { " + ln);
        sb.append("this.h = h;");
        sb.append("}" + ln);
        for (Method m : interfaces[0].getMethods()) {
            Class<?>[] params = m.getParameterTypes();
            StringBuffer paramNames = new StringBuffer();
            StringBuffer paramValues = new StringBuffer();
            StringBuffer paramClasses = new StringBuffer();
            for (int i = 0; i < params.length; i++) {
                Class clazz = params[i];
                String type = clazz.getName();
                String paramName = toLowerFirstCase(clazz.getSimpleName());
                paramNames.append(type + " " + paramName);
                paramValues.append(paramName);
                paramClasses.append(clazz.getName() + ".class");
                if (i > 0 && i < params.length - 1) {
                    paramNames.append(",");
                    paramClasses.append(",");
                    paramValues.append(",");
                }
            }
            sb.append("public " + m.getReturnType().getName() + " " + m.getName() + "(" + paramNames.toString() + ") {" + ln);
            sb.append("try{" + ln);
            sb.append("Method m = " + interfaces[0].getName() + ".class.getMethod(\"" + m.getName() + "\",new Class[]{" + paramClasses.toString() + "});" + ln);
            sb.append((hasReturnValue(m.getReturnType()) ? "return " : "") + getCaseCode("this.h.invoke(this,m,new Object[]{" + paramValues + "})", m.getReturnType()) + ";" + ln);
            sb.append("}catch(Error _ex) { }");
            sb.append("catch(Throwable e){" + ln);
            sb.append("throw new UndeclaredThrowableException(e);" + ln);
            sb.append("}");
            sb.append(getReturnEmptyCode(m.getReturnType()));
            sb.append("}");
        }
        sb.append("}" + ln);
        return sb.toString();
    }
    private static Map<Class, Class> mappings = new HashMap<Class, Class>();
    static {
        mappings.put(int.class, Integer.class);
    }
    private static String getReturnEmptyCode(Class<?> returnClass) {
        if (mappings.containsKey(returnClass)) {
            return "return 0;";
        } else if (returnClass == void.class) {
            return "";
        } else {
            return "return null;";
        }
    }
    private static String getCaseCode(String code, Class<?> returnClass) {
        if (mappings.containsKey(returnClass)) {
            return "((" + mappings.get(returnClass).getName() + ")" + code + ")." + returnClass.getSimpleName() + "Value()";
        }
        return code;
    }
    private static boolean hasReturnValue(Class<?> clazz) {
        return clazz != void.class;
    }
    private static String toLowerFirstCase(String src) {
        char[] chars = src.toCharArray();
        chars[0] += 32;
        return String.valueOf(chars);
    }
}
第三步,创建 GPClassLoader 类:
package org.example.spring.designpattern.proxy.myproxy;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
/**
 * @author ss_419
 */
public class GPClassLoader extends ClassLoader {
    private File classPathFile;
    public GPClassLoader(){
        String classPath = GPClassLoader.class.getResource("").getPath();
        this.classPathFile = new File(classPath);
    }
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        String className = GPClassLoader.class.getPackage().getName() + "." + name;
        if(classPathFile  != null){
            File classFile = new File(classPathFile,name.replaceAll("\\.","/") + ".class");
            if(classFile.exists()){
                FileInputStream in = null;
                ByteArrayOutputStream out = null;
                try{
                    in = new FileInputStream(classFile);
                    out = new ByteArrayOutputStream();
                    byte [] buff = new byte[1024];
                    int len;
                    while ((len = in.read(buff)) != -1){
                        out.write(buff,0,len);
                    }
                    return defineClass(className,out.toByteArray(),0,out.size());
                }catch (Exception e){
                    e.printStackTrace();
                }
            }
        }
        return null;
    }
}
第四步,创建 GPMeipo 类:
package org.example.spring.designpattern.proxy.myproxy;
import java.lang.reflect.Method;
/**
 * @author ss_419
 */
public class GPMeipo implements GPInvocationHandler {
    private Object target;
    public Object getInstance(Object person) throws Exception{
        this.target = person;
        Class<?> clazz = target.getClass();
        return GPProxy.newProxyInstance(new GPClassLoader(),clazz.getInterfaces(),this);
    }
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        before();
        Object obj = method.invoke(this.target,args);
        after();
        return obj;
    }
    private void before(){
        System.out.println("我是媒婆,我要给你找对象,现在已经确认你的需求");
        System.out.println("开始物色");
    }
    private void after(){
        System.out.println("OK的话,准备办事");
    }
}
第五步,客户端测试代码:
package org.example.spring.designpattern.proxy.myproxy;
import org.example.spring.designpattern.proxy.dynamicproxy.Customer;
import org.example.spring.designpattern.proxy.staticproxy.Person;
public class GPProxyTest {
    public static void main(String[] args) {
        try {
            //JDK动态代理的实现原理
            Person obj = (Person) new GPMeipo().getInstance(new Customer());
            System.out.println(obj.getClass());
            obj.findLove();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}
												
											Spring设计模式——代理模式[手写实现JDK动态代理]的更多相关文章
- JDK动态代理深入理解分析并手写简易JDK动态代理(下)
		
原文同步发表至个人博客[夜月归途] 原文链接:http://www.guitu18.com/se/java/2019-01-05/27.html 作者:夜月归途 出处:http://www.guitu ...
 - JDK动态代理深入理解分析并手写简易JDK动态代理(上)
		
原文同步发表至个人博客[夜月归途] 原文链接:http://www.guitu18.com/se/java/2019-01-03/27.html 作者:夜月归途 出处:http://www.guitu ...
 - 模式的秘密-代理模式(2)-JDK动态代理
		
代理模式-动态代理 (1) (2) 代码实践动态代理: 第一步:被代理类的接口: package com.JdkProxy; public interface Moveable { void move ...
 - 浅谈代理模式与java中的动态代理
		
代理模式的定义: 代理模式是一个使用律非常高的模式,定义如下: 为其他对象提供一种代理,以控制对这个对象的访问. 类图: 简单的静态代理: public interface IRunner{ //这是 ...
 - java代理(静态代理和jdk动态代理以及cglib代理)
		
版权声明:本文为Fighter168原创文章,未经允许不得转载. 目录(?)[+] 说到代理,脑袋中浮现一大堆代理相关的名词,代理模式,静态代理,jdk代理,cglib代理等等. 记忆特别深刻 ...
 - 做一些Spring AOP做过的事,封装 jdk动态代理成为一个黑盒子
		
怎么使用eclise 抽取方法,请看 利用eclipse 抽取代码片段为方法 抽取完成之后,还需要 ① 将Collection.class换成 target.getClass(),targ ...
 - 代理模式与java中的动态代理
		
前言 代理模式又分为静态代理与动态代理,其中动态代理是Java各大框架中运用的最为广泛的一种模式之一,下面就用简单的例子来说明静态代理与动态代理. 场景 李雷是一个唱片公司的大老板,很忙, ...
 - 10分钟了解 代理模式与java中的动态代理
		
前言 代理模式又分为静态代理与动态代理,其中动态代理是Java各大框架中运用的最为广泛的一种模式之一,下面就用简单的例子来说明静态代理与动态代理. 场景 李雷是一个唱片公司的大老板,很忙, ...
 - JDK动态代理浅析
		
原文同步发表至个人博客[夜月归途] 原文链接:http://www.guitu18.com/se/java/2018-06-29/17.html 作者:夜月归途 出处:http://www.guitu ...
 - JDK动态代理案例与原理分析
		
一.JDK动态代理实现案例 Person接口 package com.zhoucong.proxy.jdk; public interface Person { // 寻找真爱 void findlo ...
 
随机推荐
- Vue.Draggable使用总结
			
Draggable为基于Sortable.js的vue组件,用以实现拖拽功能. 特性 支持触摸设备 支持拖拽和选择文本 支持智能滚动 支持不同列表之间的拖拽 不以jQuery为基础 和视图模型同步刷新 ...
 - 暑假学习二 8.24 Hadoop的环境配置
			
今日学习内容: 主要为Hadoop的环境配置,相关配置所需文档可以留言(?)会给发送 1.hadoop介绍: 狭义: 核心组件,Hadoop hdfs 分布存储 yarn 资源管理和任务调度框架 ...
 - redhat单网卡配置多个IP
			
单网卡配置多个IP 进入配置网络的文件目录 复制网络配置文件并修改名字 配置网络 网络1 网络2 激活网络设备 重启网络 重启计算机,查看IP地址 (有问题可以提出来)
 - 【Java】无法将java.util.LinkedHashMap强制转换为X
			
JsonJack:无法将java.util.LinkedHashMap强制转换为X 1. 概述 Jackson是一个广泛使用的Java库,它使可以方便地对JSON或XML进行序列化/反序列化.有时,当 ...
 - Failed to apply plugin [id com.android.application]
			
Failed to apply plugin [id com.android.application] 这种问题解决方法很简单,在BuildOutPut里面就能找到. 解决方法:在gradle.pro ...
 - Jetpack compose学习笔记之ConstraintLayout(布局)
			
一,简介 Jetpack compose中没有提供ConstraintLayout支持,所以需要添加下面的依赖来导入. // build.gradle implementation "and ...
 - outlook2013 关闭后不能接收邮件了解决方法
			
本人装的是2013版的outlook亲测有用,其他版本的本人没试过. 下载KeepOutlookRunning.rar 链接:https://pan.baidu.com/s/1hcNorKDLbpzV ...
 - js-label
			
js中的label就像一个对已有语句块的命名,函数有了函数名我们可以随时调用它,语句块有了语句名我们也可以随时调用它,将他运用到循环中可快速跳出 循环. var num = 0;for (var i ...
 - Python+Django(2)——创建应用程序
			
新打开一个终端窗口,切换到manage.py所在的目录 激活虚拟环境:ll_env\Scripts\activate 命令startapp appname 让Django建立创建应用程序所需的基础设施 ...
 - 匿名Lambda函数,C++
			
1 // To Compile and Run: g++ -std=c++11 lambda.cc -Wall -O3 && ./a.out 2 3 4 #include <io ...