Java动态代理

代理模式

解说:给某一个对象提供一个代理,并由代理对象控制对原对象的引用;

代理模式需要以下几个角色:

1  主题:规定代理类和真实对象共同对外暴露的接口;

2  代理类:专门代理真实对象的类;

3  真实对象:需要被代理的对象;

代理解决的主要的业务就是需要在 真实对象的某个接口 前后处理一些事情,框架中多会用到这种功能,比如 打日志、记录时间等

静态代理

静态代理是指自己动手编写代码实现代理类;

优点:业务类只需要关注业务逻辑本身,保证了业务类的重用性。这是代理的共有优点。

缺点:每一个真实对象都需要一个具体的代理类,不能做到可重用;

静态代理比较简单,下边用代码来具体说明;

主题接口:IAnimal

Java动态代理_第1张图片
IAnimal

真实对象:Dog

Java动态代理_第2张图片
dog

代理:DogProxy

Java动态代理_第3张图片
dogproxy

静态代理使用:

Java动态代理_第4张图片

代理和真实对象对外暴露一致

动态代理

动态代理是指在运行时动态生成代理类;

jdk

要使用Java中原生的动态代理,需要用到以下几个类和接口

接口InvocationHandler

Proxy类

我们还是用静态代理用到的代码:主题接口IAnimal和真实对象Dog不变,去掉DogProxy和StaticProxyTest,增加以下代码

DogProxyInvocationHandler

Java动态代理_第5张图片

DynamicProxyTest

Java动态代理_第6张图片

可以看出:Java动态代理 我们必须有真实对象,实现了InvocationHandler接口的自己的处理类,然后通过Proxy生成代理类

输出如下:

这里动态代理的优势相比静态代理为:即使真实对象有N个接口,我们的invocationHandler只需要一个Invoke方法即可!

这里有几个问题:

1  动态代理生成的class name为什么是 $Proxy0 ?

Java动态代理_第7张图片

相关变量如下:

Java动态代理_第8张图片

以上便可以解决相关问题

2  动态代理生成的代理类到底是什么样子的?生成代理类的关键接口是什么?为何调用真实对象的某个接口会进入invoke方法?

通过分析源码:我们知道 Proxy.newProxyInstance → Proxy.getProxyClass0 → WeakCache.get → WeakCache.Factory.get → Proxy.ProxyClassFactory.apply → ProxyGenerator.generateProxyClass

最终生成了一个 byte[] 类型的 class类;这样byte[] 比较抽象 ,我们想看到该怎么办?可以通过下边的代码生成Proxy0

Java动态代理_第9张图片

生成的代码如下

Java动态代理_第10张图片

以上代码为我们解答了红色的问题。虽然代码是这样的  代理类集成了 Proxy类,但是如果想要验证 如何验证呢?

cglib

cglib是什么?CGLIB is a powerful, high performance code generation library.

特点简单说:

CGLib (Code Generation Library) 是一个强大的,高性能,高质量的Code生成类库;

它可以在运行期扩展Java类与实现Java接口;

CGLib 比 Java 的 java.lang.reflect.Proxy 类更强的在于它不仅可以接管接口类的方法,还可以接管普通类的方法。

CGLib 的底层是Java字节码操作框架 —— ASM

引入JAR包支持,如下:

目前最新的版本是 3.2.4,CGLib的package分布和作用如下:

net.sf.cglib.core:底层字节码处理类,他们大部分与ASM有关系,对其进行封装,更易于使用;

net.sf.cglib.transform:编译期或运行期类和类文件的转换;

net.sf.cglib.proxy:实现创建代理和方法拦截器的类;

net.sf.cglib.reflect:实现快速反射的类;

net.sf.cglib.util:集合排序工具类;

net.sf.cglib.beans:JavaBean相关的工具类;

我们沿用上边的例子,来做下演示:

加入你有一个类Animal,打算对里边的所有方法进行包装,由于这个类没有实现接口,所以你无法使用jdk 动态代理

Java动态代理_第11张图片

你现在想要在testt方法输出前后不加任何内容,但是另外两个方法输出前后要加一个字符串,效果如下:

Java动态代理_第12张图片

其中=======是分隔线,上边两个方法前后都改变了,但是最后一个方法则没做任何改变还是原生的。如何做到?

首先我们要定义一个拦截器,该拦截器实现了 cgLib的MethodInterceptor,如下:

Java动态代理_第13张图片

再定义一个拦截器过滤器,如下:

Java动态代理_第14张图片

接下来看下测试类:

Java动态代理_第15张图片

CGLib通过 Enhancer、Callback、CallbackFilter就可以实现上述功能了。

深入代码:Enhancer → KeyFactory.Generator → AbstractClassGenerator → DefaultGeneratorStrategy.generate → KeyFactory.Generator.generateClass → ClassWriter.toByteArray 生成 class bytecode .

生成的代码反编译如下:

Java动态代理_第16张图片

附博客:动态代理

你可能感兴趣的:(Java动态代理)