Java * 的多种实现方式
作者:卡修斯 发布时间:2023-11-20 05:23:14
标签:Java, ,
目录
一、 * 简介
二、 * 的多种实现
1. 基于JDK的实现
2. 基于cglib的实现
三、为什么要有基于cglib的实现
四、两种方式的适用场景
JDK *
优点
缺点
适用场景
cglib
优点
缺点
适用场景
一、 * 简介
优势:在不修改源码的情况下,对目标方法进行相应的增强。
作用:完成程序功能之间的松耦合。
二、 * 的多种实现
JDK代理:基于接口的 * 技术(缺点,目标对象必须有接口,如果没有接口,则无法完成 * 的实现)
cglib代理:基于父类的 * 技术
两者的区别如图所示:
1. 基于JDK的实现
目标接口类:
public interface TargetInterface {
public void save();
public void print(String str);
}
目标类:
public class Target implements TargetInterface{
public void save() {
System.out.println("save running...");
}
public void print(String str) {
System.out.println(str);
}
}
增强类:
public class Advice {
public void before() {
System.out.println("前置增强");
}
public void after() {
System.out.println("后置增强");
}
}
测试类:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyTest {
public static void main(String[] args) {
//目标对象
final Target target = new Target();
//增强对象
final Advice advice = new Advice();
TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance(
target.getClass().getClassLoader(), //目标对象类加载器
target.getClass().getInterfaces(), //目标对象相同的接口字节码对象数组
new InvocationHandler() {
//调用代理对象的任何方法,实质执行的都是invoke方法
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
advice.before();//前置增强
Object invoke = method.invoke(target, args);//执行目标方法
advice.after();//后置增强
System.out.println();
return invoke;
}
});
//代理对象的方法测试
proxyInstance.save();
proxyInstance.print("JDK * ");
}
}
运行截图:
2. 基于cglib的实现
需要导入Jar包,如果是maven项目,则在pom.xml文件加入如下配置:
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>4.2.4.RELEASE</version>
</dependency>
目标类:
public class Target {
public void save() {
System.out.println("save running...");
}
public void print(String str) {
System.out.println(str);
}
}
增强类:
public class Advice {
public void before() {
System.out.println("前置增强");
}
public void after() {
System.out.println("后置增强");
}
}
测试类:
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
public class ProxyTest {
public static void main(String[] args) {
final Target target = new Target();
final Advice advice = new Advice();
//返回值就是动态生成的代理对象,基于cglib
//创建增强器
Enhancer enhancer = new Enhancer();
//设置父类(目标)
enhancer.setSuperclass(Target.class);
//设置回调
enhancer.setCallback(new MethodInterceptor() {
public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{
advice.before();
Object invoke = method.invoke(target, obj);
advice.after();
System.out.println();
return invoke;
}
});
//创建代理对象
Target proxy = (Target)enhancer.create();
//测试代理方法
proxy.save();
proxy.print("基于cglib实现动态规划");
}
}
运行截图:
三、为什么要有基于cglib的实现
使用JDK * 实现时,最大限制是被增强对象必须实现接口,并且增强的方法只能是接口中声明的方法。但在实际的项目中,可能总是存在对不实现业务接口的对象进行增强的需求,这时JDK * 将无能为力。
四、两种方式的适用场景
JDK *
优点
不依赖第三方jar包, 使用方便
随着JDK的升级,JDK * 的性能在稳步提升
缺点
只能代理实现了接口的类
执行速度较慢
适用场景
如果你的程序需要频繁、反复地创建代理对象,则JDK * 在性能上更占优。
cglib
优点
由于是动态生成字节码实现代理,因此代理对象的执行速度较快, 约为JDK * 的1.5 ~ 2倍
可以代理没有实现接口的对象
缺点
不能代理final类
动态生成字节码虽然执行较快,但是生成速度很慢,根据网上一些人的测试结果,cglib创建代理对象的速度要比JDK慢10 ~ 15倍。
适用场景
不需要频繁创建代理对象的应用,如Spring中默认的单例bean,只需要在容器启动时生成一次代理对象。
来源:https://juejin.cn/post/6970512671368019981


猜你喜欢
- 一、List<T>对象中的T是值类型的情况(int 类型等)对于值类型的List直接用以下方法就可以复制:List<T&g
- springboot生成bean名称冲突问题描述我们再使用springboot的时候,在不同的文件目录下,可能存在相同名称的java类,这个
- stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,瞬
- mapper.xml中if标签test判断的用法1. 字符串等于条件的两种写法① 将双引号和单引号的位置互换<if test='
- 1 二叉排序树的概述本文没有介绍一些基础知识。对于常见查找算法,比如顺序查找、二分查找、插入查找、斐波那契查找还不清楚的,可以看这篇文章:常
- 本文参考借鉴:https://www.jb51.net/article/102983.htm先上效果图:自定义控件:AttendancePr
- 前言最近做公司项目的时候,经常会遇到一个问题,就是我为某个控件如EditText设置requestfocus()的时候不管用,比如说登陆的时
- 本文实例讲述了C#计算矩阵的逆矩阵方法。分享给大家供大家参考。具体如下:1.代码思路1)对矩阵进行合法性检查:矩阵必须为方阵2)计算矩阵行列
- 在算法面试中,面试官总是喜欢围绕链表、排序、二叉树、二分查找来做文章,而大多数人都可以跟着专业的书籍来做到倒背如流。而面试官并不希望招收的是
- 本文实例讲述了C#对list列表进行随机排序的方法。分享给大家供大家参考。具体实现方法如下:public List<T> Ran
- 这里文章写出来并不是为了炫耀什么,只是觉得发现些好东西就分享出来而已,同时也做个记录,方便以后查找开始正文1、先介绍本文会用到的window
- 背景由于前前前阵子写了个壳,得去了解类的加载流程,当时记了一些潦草的笔记。这几天把这些东西简单梳理了一下,本文分析的代码基于Android8
- 本文实例为大家分享了C#实现验证码功能的具体代码,供大家参考,具体内容如下分析需要四个字符(字母(大小写)+数字)将四个字符连接成字符串将连
- 首先给出一段代码:public class AslistMethod { public static void main(String[]
- 本文实例为大家分享了JavaWeb实现注册用户名检测的具体代码,供大家参考,具体内容如下案例说明实现一个可以异步获取用户名是否被注册的小案例
- 不论你是否关注,Java Web应用都或多或少的使用了线程池来处理请求。线程池的实现细节可能会被忽视,但是有关于线程池的使用和调优迟早是需要
- 这两个类使用起来非常方便,可以完成我们对定时器的绝大多数需求Timer类是用来执行任务的类,它接受一个TimerTask做参数Timer有两
- 本文实例讲述了Android结束进程的方法。分享给大家供大家参考,具体如下:最近在做一个类似与任务管理器的东西,里面有个功能,可以通过这个管
- 本文实例讲述了C#实现左截取和右截取字符串的方法,分享给大家供大家参考。具体方法分析如下:问题如下:使用C#语法编写程序时,我们需要截取一个
- 我也不知道这个叫什么,就是比如我要打开我电脑的计算机,可以直接在命令行输入“calc”就可以了。现在用让代码去执行。public stati