Java 学习笔记

👉 所有文章
基础 Java 安装 Java : 第一个程序 Hello World Java : 建议使用 UTF-8 编写 Java 代码 Java : package 包命名规范 使用 Intellij IDEA 创建 Java 项目 Java 布尔类型 Java 处理日期和时间 Java 正则表达式 Java finalize 方法 Java : 空值 null Java 如何触发垃圾回收 Java ThreadLocal Java InheritableThreadLocal Java Integer之间的比较 Java 动态代理 Java 匿名类 Java 枚举 Java: 加速maven、gradle依赖下载 Java 如何静态导入 import static println Java 引用级别:强引用、软引用、弱引用、幽灵引用 Java try finally return 解惑 Java WeakHashMap Java ReferenceQueue 怎么写 Java 示例代码? Java 匿名类双大括号初始化 什么是 Java Bean Java 多行字符串 Java 快速生成 List Java 快速生成 Map Java 将异常堆栈转换为 String JDK SPI 的使用和源码分析 Java Map 中的 key 和 value 能否为 null Java List 和 数组的互相转换 Java 获取环境变量 Java 获取和设置系统属性 Java: 如何获取当前进程的 PID ? Java 字符串左侧/右侧补充空格或者其他字符 Java 线程 Java: 如何获取文本文件内容 Java: 读取资源文件内容 Java: 获取 jar 中文件的内容 Java: 使用 JavaFx 构建 GUI Java: Class 类 Java : 使用 instanceof 判断对象类型 一个自定义的 Java 工具类 Java : 获取当前函数所属类的类名 Java : 获取当前执行的函数名 Java : 使用 String 的 split 函数拆分字符串 Java : 获取字符的 Unicode 编号(代码点) Java : 获取当前工作目录 Java : 使用 Class 对象的 isArray 方法判断对象是否为数组 使用 Java 生成 CSV 文件
工具 Java : jps 命令的使用 Java : jcmd 命令的使用 Java : VisualVM 工具的使用 Java : 使用 javap 解析 class 文件 Java : jar 命令
项目构建 ( 基于 Gradle ) Java: 使用 Gradle 将源码打包为 jar Java: Gradle 下载项目依赖 Java : 清理 Gradle 生成的 build、out 目录 Java : 将 maven 项目转换为 gradle 项目
测试 Java Mockito 测试框架快速入门 JMockit 入门 JUnit 入门 JUnit 单测隔离
三方库 Java JOOR 反射库 Java alibaba transmittable-thread-local 库:让 ThreadLocal 跨线程传播 Java 日志组件 slf4j 的使用和源码分析 Java Lombok 库:为你减少样板代码 Java: 使用 cglib 实现动态代理 Java Hibernate validator 校验框架 Java 使用 Hessian2 序列化和反序列化 H2 数据库快速入门 Java : 使用 Gson 库处理json数据
IDE Intellij IDEA 使用技巧汇总 Intellij IDEA 如何修改/添加代码颜色主题 解决 Intellij IDEA 右下角不显示 git 分支的问题 Intellij IDEA 指定/修改 JDK 版本 Intellij IDEA 文件初始化模板
其他 Java: 加速maven、gradle依赖下载 Java: 如何创建多模块项目 Java 集成 groovy 构建规则引擎 Java 13: 安装 Java 13 新特性:文本块(多行字符串) 卸载 MacOS 上安装的 Java Java: 执行 sql 文件 Java JDK 有哪些发行版 ? java拾遗:String和数组 java拾遗:由反转数组想到System.out的实现机制 java拾遗:如何读取properties文件内容 Java并发概念汇总 java拾遗:System.out.println()是什么? java拾遗:通过示例理解位运算 使用“庖丁解牛”进行中文分词 DBUtils简明教程 试用velocity模板引擎
TODO Java: 如何使用 java、javac 、jar 命令 使用 Nexus Repository Manager 搭建 maven 私有仓库

Java : 使用 cglib 实现动态代理


cglib 项目地址:https://github.com/cglib/cglib

Java自带的动态代理不同,cgilib 动态代理没有必须使用接口的限制。也就是,如果一个类没有实现任何接口,cglib 支持对其生成动态代理。当然,如果一个类实现了一些接口,cglib 也支持对其生成动态代理。cglib 的代理机制基于继承。

要使用 cglib,需要引入依赖。如果使用 gradle 构建项目,配置依赖如下:

compile group: 'cglib', name: 'cglib', version: '3.2.10'

入门

Hello 类是要被代理的类:

package demo01;

public class Hello {

    public void hello() {
        System.out.println("hello");
    }

    public void hi() {
        System.out.println("hi");
    }

}

方法执行的拦截器类,用于实现动态代理:

package demo01;

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

import java.lang.reflect.Method;

public class CustomInterceptor implements MethodInterceptor {

    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.printf("发起调用. obj: %s, method: %s, proxy: %s\n",
                obj.getClass().getCanonicalName(),
                method.getName(),
                proxy.getClass().getCanonicalName()
        );
        return proxy.invokeSuper(obj, args);
    }

    // 该函数用于生成代理
    public static Object createProxy(Class cls) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(cls);
        enhancer.setCallback(new CustomInterceptor());
        return enhancer.create();
    }

}

测试代理效果:

package demo01;

public class Main {

    public static void main(String[] args) {
        Hello hello = (Hello) CustomInterceptor.createProxy(Hello.class);
        hello.hello();
    }

}

运行结果:

发起调用. obj: demo01.Hello$$EnhancerByCGLIB$$564008dc, method: hello, proxy: net.sf.cglib.proxy.MethodProxy
hello

MethodInterceptor 是方法级别的代理

上面的示例中,CustomInterceptor 继承自 MethodInterceptor 。MethodInterceptor 实现的是方法级别的代理。当被代理类进行自调用时,依然会执行代理逻辑。

在 Hello 类的 hello 方法中增加自调用:

public class Hello {

    public void hello() {
        System.out.println("hello");
        hi(); // 调用 hi
    }

    public void hi() {
        System.out.println("hi");
    }

}

运行下面的测试代码:

public class Main {

    public static void main(String[] args) {
        Hello hello = (Hello) CustomInterceptor.createProxy(Hello.class);
        hello.hello();
    }

}

结果如下:

发起调用. obj: demo02.Hello$$EnhancerByCGLIB$$263b7947, method: hello, proxy: net.sf.cglib.proxy.MethodProxy
hello
发起调用. obj: demo02.Hello$$EnhancerByCGLIB$$263b7947, method: hi, proxy: net.sf.cglib.proxy.MethodProxy
hi

使用 InvocationHandler 构造类级别的代理

有时候,我们并不想自调用的时候走代理逻辑,那么,可以使用 InvocationHandler 构造类级别的代理。

Java本身的动态代理中也涉及一个类,名字也是 InvocationHandler 。不过它的全路径是 java.lang.reflect.InvocationHandler。 cglib 中的全路径是 net.sf.cglib.proxy.InvocationHandler。

我们看下如何使用:

package demo03;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.InvocationHandler;

import java.lang.reflect.Method;

public class CustomInterceptor implements InvocationHandler {

    private Object target;

    public CustomInterceptor(Object target) {
        this.target = target;
    }

    public static Object createProxy(Object target) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(target.getClass());
        enhancer.setCallback(new CustomInterceptor(target));
        return enhancer.create();
    }


    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        System.out.printf("发起调用. method: %s, proxy: %s\n",
                method.getName(),
                proxy.getClass().getCanonicalName()
        );

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

假如被代理类代码是:

package demo03;

public class Hello {

    public void hello() {
        System.out.println("hello");
        hi();
    }

    public void hi() {
        System.out.println("hi");
    }

}

测试代理效果:

package demo03;

public class Main {

    public static void main(String[] args) {
        Hello hello = (Hello) CustomInterceptor.createProxy(new Hello());
        hello.hello();
    }

}

运行结果如下:

发起调用. method: hello, proxy: demo03.Hello$$EnhancerByCGLIB$$8913a7da
hello
hi

可以看到,输出 hi 时,没有输出 CustomInterceptor 中那些额外内容。


( 本文完 )

文章目录