利用代码分别实现jdk动态代理和cglib动态代理_面试之动态代理

1a4e2672fd8bb4fdbff91d11a9d55259.png

大家好!我是CSRobot,从今天开始,我将会发布一些技术文章,内容就是结合春招以来的面试所遇到的问题进行分享,首先会对知识点进行一个探讨和整理,在最后会给出一些面试题并作出解答,希望可以帮助到大家!今天知识点是动态代理,会分从以下几个方面进行探讨:

  • 静态代理
  • JDK动态代理
  • Cglib动态代理
  • 面试题分享

在java中,要说到动态代理,那么首先需要聊聊静态代理

一、静态代理

静态代理是代理类在编译期间就创建好了,不是编译器生成的代理类,而是手动创建的类。在编译时就已经将接口,被代理类,代理类等确定下来。以下以一个例子来说明静态代理的实现

1、实现步骤

  • 创建服务类接口:BuyHouse
//创建服务接口
public interface BuyHouse {void buyHouse();
}
  • 实现服务接口: BuyHouseImpl
//实现接口
public class BuyHouseImpl implements BuyHouse {@Overridepublic void buyHouse() {System.out.println("我要买房子!");}
}
  • 创建代理类实现服务接口,在代理类中手动增强: BuyHouseProxy
public class BuyHouseProxy implements BuyHouse {private BuyHouse buyHouse;public BuyHouseProxy(final BuyHouse buyHouse) {this.buyHouse = buyHouse;}//手动增强buyHouse方法@Overridepublic void buyHouse() {System.out.println("买房前准备");buyHouse.buyHouse();System.out.println("买房后装修");}
}
  • 创建测试类
@Test
public void test() {//未增强的类BuyHouse buyHouse = new BuyHouseImpl();buyHouse.buyHouse();//创建代理类BuyHouseProxy buyHouseProxy = new BuyHouseProxy(buyHouse);buyHouseProxy.buyHouse();
}
  • 输出:
我要买房子!
买房前准备
我要买房子!
买房后装修

二、动态代理

​ Java动态代理的优势是实现无侵入式的代码扩展,也就是方法的增强;让你可以在不用修改源码的情况下,增强一些方法;在方法的前后你可以做你任何想做的事情。

​ 在java的动态代理中,主要有两种实现方式,首先是基于反射的JDK动态代理实现;其次是cglib的字节码技术的实现

1、JDK动态代理

  • 通过实现 InvocationHandler 接口创建自己的调用处理器
  • 通过为 Proxy 类指定 ClassLoader 对象和一组 interface 来创建动态代理类
  • 通过反射机制获得动态代理类的构造函数,其唯一参数类型是调用处理器接口类型
  • 通过构造函数创建动态代理类实例,构造时调用处理器对象作为参数被传入
  • 代理类名称是包名+$Proxy+id序号
  • JDK动态代理只能代理接口而不能代理类,原因大家可以根据jdk动态代理的实现步骤大胆推断一下,后文会给出答案
  • 创建服务接口
//创建服务接口
public interface BuyHouse {void buyHouse();
}public class BuyHouseImpl implements BuyHouse {@Overridepublic void buyHouse() {System.out.println("我要买房子!");}
}
  • 编写动态处理器
public class DynamicProxyHandler implements InvocationHandler{private Object object;public DynamicProxyHandler(final Object object) {this.object = object;}//invoke方法对业务进行增强@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("买房前准备");Object result = method.invoke(object, args);System.out.println("买房后装修");return result;}public Object getProxyInstance(){return Proxy.newProxyInstance(object.getClass().getClassLoader(),                object.getClass().getInterfaces(), this);}
}
  • 测试动态代理
@Testpublic void test1() {//创建需要被代理的类BuyHouse buyHouse = new BuyHouseImpl();//创建动态代理处理器DynamicProxyHandler DP = new DynamicProxyHandler(buyHouse);//获得代理类BuyHouse proxyBuyHouse = DP.getProxyInstance();proxyBuyHouse.buyHouse();}
  • 输出
买房前准备
我要买房子!
买房后装修

2、Cglib动态代理

在cglib中提供了一个Enhance类来创建代理类,类似于jdk动态代理中的Proxy类。

  • 添加maven依赖包
<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>2.2.2</version>
</dependency>
  • 同样是使用上文中buyHouse的类
//创建服务接口
public interface BuyHouse {void buyHouse();
}public class BuyHouseImpl implements BuyHouse {@Overridepublic void buyHouse() {System.out.println("我要买房子!");}
}
  • 创建方法拦截器
public class CglibProxy implements MethodInterceptor{private Object target;public Object getInstance(final Object target) {this.target = target;Enhancer enhancer = new Enhancer();enhancer.setSuperclass(this.target.getClass());enhancer.setCallback(this);return enhancer.create();}@Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("买房前准备");Object result = proxy.invokeSuper(obj, args);System.out.println("买房后装修");return result;}
}
  • 测试
@Test
public void test3(){BuyHouse buyHouse = new BuyHouseImpl();CglibProxy cglibProxy = new CglibProxy();BuyHouseImpl buyHouseCglibProxy = (BuyHouseImpl)cglibProxy.getInstance(buyHouse);buyHouseCglibProxy.buyHouse();
}
  • 最终输出结果和之前相同

三、面试题

1、动态代理是什么?

动态代理:为其他对象提供一个代理以控制对某个对象的访问。代理类主要负责为委托了(真实对象)预处理消息、过滤消息、传递消息给委托类,代理类不现实具体服务,而是利用委托类来完成服务,并将执行结果封装处理

2、为什么不使用静态代理而使用动态代理?

通过前文大家也可以理解了,静态代理是代理类在编译期间就创建好了,不是编译器生成的代理类,需要手动创建的类。在编译时就已经将接口,被代理类,代理类等确定下来。如果我们有很多类很多接口需要代理,那么我们就只能使用代码提前写死,不够灵活;使用了动态代理之后,我们不需要手动创建代理类,全部交给代理去完成对代理类的创建,实现无侵入式的代码扩展,这也是符合面向对象编程原则的操作。

3、JDK动态代理为什么只能代理接口而不能代理类?

通过JDK动态代理实现步骤我们就可以看到,我们通过Proxy类的newProxyInstance方法来生成代理对象,代理类继承了Proxy类并且实现了要代理的接口,由于java不支持多继承,所以JDK动态代理不能代理类

4、JDK动态代理和CgLib动态代理的区别?

1)JDK动态代理只能代理接口而不能代理类,CgLib对于接口和类都可以实现代理

2)JDK动态代理底层使用反射的方式实现;而CgLib采用了非常底层的字节码技术,其原理是通过目标类的字节码为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。代理类将目标类作为自己的父类并为其中的每个非final委托方法创建两个方法

9dbee68bbf737c6597b86495a025ed27.png

f7fb65164a918952daee4ada3e88860e.png
CSRobot

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/562661.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

将下列数组中奇数和偶数分别存放于两个不同的两个数组

将下列数组中奇数和偶数分别存放于两个不同的两个数组 /* * 将下列数组中奇数和偶数分别存放于两个不同的两个数组 * */ public class Test09 {public static void main(String[] args) {int[] arr {12,34,23,47,72,84,22,28,45,57,91};//用于统计偶数元素个数int o 0;for (…

数组工具类的使用

数组工具类的使用 package demo02; /* * 数组工具类的使用 * 数组的复制操作 * */ import java.util.*;public class Test10 {public static void main(String[] args) {int[] arr {3,4,2,66,75,4,322,55,33,85,39};System.out.println(Arrays.toString(arr));//排序问题Arra…

从键盘上录入学生人数和每个学生的分数,按分数降序输出所有的分数,java冒泡排序应用

从键盘上录入学生人数和每个学生的分数&#xff0c;按分数降序输出所有的分数 import java.util.Arrays; import java.util.Scanner;/* * 从键盘上录入学生人数和每个学生的分数&#xff0c;按分数降序输出所有的分数 * java冒泡排序 * */ public class Test12 {public static…

jvm内存结构_JVM系列之内存结构

JVM的内存结构大概分为&#xff1a;堆&#xff08;Heap&#xff09;&#xff1a;线程共享。所有的对象实例以及数组都要在堆上分配。回收器主要管理的对象。堆内存是JVM中最大的一块由年轻代和老年代组成&#xff0c;而年轻代内存又被分成三部分&#xff0c;Eden空间、From Sur…

java从键盘上录入学生人数和每个学生的姓名以及分数,按照分数降序输出,学生名次、学生姓名、学生分数

java从键盘上录入学生人数和每个学生的姓名以及分数&#xff0c;按照分数降序输出&#xff0c;学生名次、学生姓名、学生分数 import java.util.Scanner;/* * 从键盘上录入学生人数和每个学生的姓名以及分数&#xff0c;按照分数降序输出&#xff0c;学生名次、学生姓名、学生…

python label background设置成透明_纯Python绘制满满艺术感的山脊地图

点击上方"蓝字"关注我们Python大数据分析记录 分享 成长添加微信号"CNFeffery"加入技术交流群❝本文示例代码及附件已上传至我的Github仓库https://github.com/CNFeffery/DataScienceStudyNotes❞1 简介下面的这幅图可能很多读者朋友们都看到过&#x…

java二维数组的遍历

java二维数组的遍历 /* * java二维数组的遍历 * 二维数组&#xff0c;数组座位数组元素类型的数组&#xff0c;数组的数组 * */ public class Test14 {public static void main(String[] args) {int[][] arr {{1,2,3},{4,5},{6,7,8,9}};//从arr中取出每个元素//for循环遍历fo…

layout布局_安卓最常见的几种布局

Android中常用的5大布局方式有以下几种&#xff1a;线性布局&#xff08;LinearLayout&#xff09;&#xff1a;按照垂直或者水平方向布局的组件帧布局&#xff08;FrameLayout&#xff09;&#xff1a;组件从屏幕左上方布局组件表格布局&#xff08;TableLayout&#xff09;&a…

java二维数组存储数据,从键盘上录入学生人数,考试科目数,以及每个学生每科分数,输出每个学生的最高分、最低分、总分、平均分

java二维数组存储数据&#xff0c;从键盘上录入学生人数&#xff0c;考试科目数&#xff0c;以及每个学生每科分数&#xff0c;输出每个学生的最高分、最低分、总分、平均分 import java.util.Scanner;/* * 二维数组存储数据 * 从键盘上录入学生人数&#xff0c;考试科目数&am…

御用导航提示提醒_四维图新推出商用车专用导航“货车通”

导航软件已经成为人们生活中不可或缺的一部分&#xff0c;它能帮助我们能更精准、更省时的规划路线&#xff0c;然而&#xff0c;对于商用车司机来说&#xff0c;更多的行车限制让现有的手机软件远远无法满足商用车的导航需求。近日&#xff0c;四维图新发布了一款专为解决商用…

openlayers3 根据经纬度 自动画框_用这软件,让你的电脑自动搞黄色

事情是这样&#xff0c;我最近盯着电脑屏幕的时间经常超过 10 个小时&#xff0c;所以需要一款护眼软件缓解眼部疲劳。但无论是 Windows 自带的夜间模式&#xff0c;还是一些 PC 品牌推出的护眼小工具&#xff0c;功能都比较简陋&#xff0c;无法满足我工作和日常所需。我到网上…

java自定义一个方法,用于返回两个整数的和

java自定义一个方法&#xff0c;用于返回两个整数的和 /*** 自定义一个方法* 用于返回两个整数的和*/ public class Test17 {public static int getSum(int a,int b){System.out.println("2getSum方法执行了");int sum ab;System.out.println("3getSum计算的结…

java定义一个方法,向控制台输出99乘法表

java定义一个方法&#xff0c;向控制台输出99乘法表 /*** 定义一个方法* 向控制台输出99乘法表*/ public class Test18 {public static void main(String[] args) {System.out.println("1main方法执行了");print99();System.out.println("3print99方法执行结束…

java定义一个方法,向控制台输出一个整数的阶乘

java定义一个方法&#xff0c;向控制台输出一个整数的阶乘 /*** 定义一个方法&#xff0c;向控制台输出一个整数的阶乘*/ public class Test19 {//输出阶乘的方法public static void factorial(int num){int fac 1;for (int i num;i>1;i--){fac*i;}System.out.println(nu…

java定义一个方法,返回a的b次方

java定义一个方法&#xff0c;返回a的b次方 /*** 定义一个方法* 返回a的b次方*/ public class Test20 {public static long pow(int a,int b){long p 1;for (int i 1;i<b;i){p*a;}return p;}public static void main(String[] args) {int a 2;int b 3;long p pow(a,b)…

页眉页脚怎么单独设置某一页里面的_Word小技巧:如何从任意页面开始设置页眉页脚...

一般情况下&#xff0c;在文档中插入页眉页脚&#xff0c;页眉页脚会在文档的每一页都显示。但很多时候&#xff0c;文章的封面页、目录页都是不需要诸如页码类的页眉页脚的。这时候&#xff0c;难道要一页一页地修改吗&#xff1f;不需要。例&#xff1a;点击页眉处&#xff0…

java定义一个方法,返回一个浮点数保留两位小数,考虑四舍五入的结果

java定义一个方法&#xff0c;返回一个浮点数保留两位小数&#xff0c;考虑四舍五入的结果 /*** 定义一个方法* 返回一个浮点数保留两位小数* 考虑四舍五入的结果*/ public class Test21 {public static double keep2(double d){d*100;d0.5;int x (int)(d);return x/100.0;}p…

java定义一个方法,返回一个整数数组的和

java定义一个方法&#xff0c;返回一个整数数组的和 /*** 定义一个方法* 返回一个整数数组的和*/ public class Test22 {//定义数组public static int getArraySum(int[] arr){int sum 0;for (int i 0; i < arr.length; i) {sumarr[i];}return sum;}public static void m…

后端在插入数据发现重复如何正确的弹出警告_前百度面试官整理的——Java后端面试题(一)...

List 和 Set 的区别 List , Set 都是继承自 Collection 接口 List 特点&#xff1a;元素有放入顺序&#xff0c;元素可重复 &#xff0c;Set 特点&#xff1a;元素无放入顺序&#xff0c;元素不可重复&#xff0c;重复元素会覆盖掉&#xff0c;(元素虽然无放入顺序&#xff0c;…

java定义一个方法,返回整数数组的元素最大值

java定义一个方法&#xff0c;返回整数数组的元素最大值 /*** 定义一个方法* 返回整数数组的元素最大值*/ public class Test23 {public static int maxArray(int[]arr){int max arr[0];for (int i 0; i < arr.length; i) {if (max<arr[i]){max arr[i];}}return max;…