Java静态代理和动态代理的使用及原理解析

静态代理的实现

操作接口:

public interface Operate {

void doSomething();

}

操作者:

public class Operator implements Operate {

@Override

public void doSomething() {

System.out.println(“I’m doing something”);

}

}

代理者:

public class OperationProxy implements Operate {

private Operator operator = null;

@Override

public void doSomething() {

beforeDoSomething();

if(operator == null){

operator = new Operator();

}

operator.doSomething();

afterDoSomething();

}

private void beforeDoSomething() {

System.out.println(“before doing something”);

}

private void afterDoSomething() {

System.out.println(“after doing something”);

}

}

调用者:

public class StaticProxyTest {

public static void main(String[] args) {

Operate operate = new OperationProxy();//使用OperationProxy代替Operator

operate.doSomething(); //代理者代替真实者做事情

}

}

静态代理的局限性

可以看到,静态代理让调用者不用再直接持有操作者的引用,而是将一切操作交由代理者去完成。但是静态代理也有它的局限性:

  1. 如果需要增加一个需要代理的方法,代理者的代码也必须改动进而适配新的操作;

  2. 如果需要代理者代理另外一个操作者,同样需要对代理者进行扩展并且更加麻烦。

可能有人想到可以用策略模式和工厂模式分别解决上面两个问题,但是,有没有更加巧妙的方法呢?首先,我们了解一下 Java 代码的执行过程。

理解 Java 代码执行流程

要从根本上理解动态代理的实现原理,得先从 Java 代码的执行流程说起:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

JVM 在运行 .class 文件之前,首先通过 ClassLoader 将 .class 文件以二进制的形式解析并生成实例以供调用,我们的代码执行逻辑是在 JVM 的运行期系统中进行工作的,那么,我们可不可以在自己的代码里面按照 .class 的格式生成自己的 .class 文件,进而调用自定义的 ClassLoader 将其加载出来呢?答案是肯定的,这样我们就可以动态地创建一个类了。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

生成自己的 .class 文件

当然我们不用手动去一点一点拼装 .class 文件,目前比较常用的字节码生成工具有 ASMJavassist,根据这个思路,生成 .class 文件的过程如下:

import javassist.ClassPool;

import javassist.CtClass;

import javassist.CtMethod;

import javassist.CtNewMethod;

public class Test {

public static void main(String[] args) throws Exception {

ClassPool pool = ClassPool.getDefault();

//创建 AutoGenerateClass 类

CtClass cc= pool.makeClass(“com.guanpj.AutoGenerateClass”);

//定义 show 方法

CtMethod method = CtNewMethod.make(“public void show(){}”, cc);

//插入方法代码

method.insertBefore(“System.out.println(“I’m just test generate .class file by javassit…”);”);

cc.addMethod(method);

//保存生成的字节码

cc.writeFile(“D://temp”);

}

}

生成的 .class 文件如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

反编译后查看内容:

//

// Source code recreated from a .class file by IntelliJ IDEA

// (powered by Fernflower decompiler)

//

package com.guanpj;

public class AutoGenerateClass {

public void show() {

System.out.println(“I’m just test generate .class file by javassit…”);

}

public AutoGenerateClass() {

}

}

可以看到,javassit 生成的类中,除了 show() 方法之外还默认生成了一个无参的构造方法。

自定义类加载器加载

为了能够让自定的类被加载出来,我们自定义了一个类加载器来加载指定的 .class 文件:

public class CustomClassLoader extends ClassLoader {

public CustomClassLoader() {

}

protected Class<?> findClass(String className) {

String path = “D://temp//” + className.replace(“.”,“//”) + “.class”;

byte[] classData = getClassData(path);

return defineClass(className, classData, 0, classData.length);

}

private byte[] getClassData(String path) {

try {

InputStream ins = new FileInputStream(path);

ByteArrayOutputStream baos = new ByteArrayOutputStream();

int bufferSize = 4096;

byte[] buffer = new byte[bufferSize];

int bytesNumRead = 0;

while ((bytesNumRead = ins.read(buffer)) != -1) {

baos.write(buffer, 0, bytesNumRead);

}

return baos.toByteArray();

} catch (IOException e) {

e.printStackTrace();

}

return null;

}

}

接着,用 ClassLoader 加载刚才生成的 .class 文件:

public class TestLoadClass {

public static void main(String[] args) throws Exception {

CustomClassLoader classLoader = new CustomClassLoader();

Class clazz = classLoader.findClass(“com.guanpj.AutoGenerateClass”);

Object object = clazz.newInstance();

Method showMethod = clazz.getMethod(“show”, null);

showMethod.invoke(object, null);

}

}

后台输出如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

成功执行了 show 方法!

利用 JDK 中的 Proxy 类进行动态代理

使用动态代理的初衷是简化代码,不管是 ASM 还是 Javassist,在进行动态代理的时候操作还是不够简便,这也违背了我们的初衷。我们来看一下怎么 InvocationHandler 怎么做:

InvocationHandler:

public class InvocationHandlerImpl implements InvocationHandler {

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

写在最后

为了这次面试,也收集了很多的面试题!

以下是部分面试题截图

Java程序员秋招三面蚂蚁金服,我总结了所有面试题,也不过如此
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!**

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

写在最后

为了这次面试,也收集了很多的面试题!

以下是部分面试题截图

[外链图片转存中…(img-RfUofas8-1713202114123)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值