Java反射(JDK)與動(dòng)態(tài)代理(CGLIB)詳解
一、反射
概念:在運(yùn)行狀態(tài)中,對(duì)于任意的一個(gè)類,都能夠知道這個(gè)類的所有字段和方法,對(duì)任意一個(gè)對(duì)象都能夠通過反射機(jī)制調(diào)用一個(gè)類的任意方法
實(shí)現(xiàn)方法:JVM在第一次加載某個(gè)類時(shí)會(huì)生成一個(gè)Class對(duì)象,里面記錄了這個(gè)類的信息
鏈接:類加載機(jī)制(留坑)
二、動(dòng)態(tài)代理
動(dòng)態(tài)代理的作用:在不改變?cè)a的基礎(chǔ)上增加新的功能,如日志、權(quán)限檢驗(yàn)等
反射在動(dòng)態(tài)代理中的應(yīng)用:由于知道原類的字段、方法等信息,才可以通過代理類執(zhí)行被代理類的方法
動(dòng)態(tài)代理的實(shí)現(xiàn)有兩種
1、JDK代理
實(shí)現(xiàn)方法:通過創(chuàng)建一個(gè)代理類,這個(gè)代理類繼承于一個(gè)Proxy類,Proxy類中有一個(gè)InvocationHandler接口,這個(gè)接口持有被代理對(duì)象和一個(gè)invoke()方法。創(chuàng)建好代理類對(duì)象后,對(duì)該對(duì)象調(diào)用的方法都會(huì)交由invoke方法處理。invoke方法接受3個(gè)參數(shù):代理對(duì)象、方法、參數(shù)列表。重寫invoke方法便可以在原方法的基礎(chǔ)上添加其他邏輯
一個(gè)JDK代理的簡(jiǎn)單實(shí)現(xiàn):
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; // 定義一個(gè)接口 interface MyInterface { void fun(); } // 用一個(gè)類去實(shí)現(xiàn)這個(gè)接口 class Person implements MyInterface { @Override public void fun() { System.out.println("Person實(shí)現(xiàn)接口方法"); } } // 用一個(gè)類實(shí)現(xiàn)InvocationHandler接口 class MyInvocationHandler<T> implements InvocationHandler { //InvocationHandler持有的被代理對(duì)象 T target; public MyInvocationHandler(T target) { this.target = target; } /** * proxy:代表動(dòng)態(tài)代理對(duì)象 * method:代表正在執(zhí)行的方法 * args:代表調(diào)用目標(biāo)方法時(shí)傳入的實(shí)參 */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("代理執(zhí)行" + method.getName() + "方法"); // 在原方法的基礎(chǔ)上添加其他邏輯 Object result = method.invoke(target, args); // 通過invoke方法調(diào)用原方法 return result; } } public class Main { public static void main(String[] args) { Person person = new Person(); InvocationHandler invocationHandler = new MyInvocationHandler<>(person); MyInterface proxy = (MyInterface) Proxy.newProxyInstance(Person.class.getClassLoader(), Person.class.getInterfaces(), invocationHandler); proxy.fun(); } }
輸出結(jié)果:
代理執(zhí)行fun方法
Person實(shí)現(xiàn)接口方法
缺陷:只能代理接口方法,因?yàn)镴DK代理需要繼承一個(gè)Proxy類,又由于Java的單繼承機(jī)制,導(dǎo)致代理類無法繼承父類的函數(shù),只能實(shí)現(xiàn)接口
2、CGLIB代理
原理與JDK代理類似,區(qū)別在于CGLIB代理創(chuàng)建的代理類直接繼承于被代理類,所以可以實(shí)現(xiàn)被代理類的方法而非僅僅接口方法
一個(gè)簡(jiǎn)單的CGLIB代理實(shí)現(xiàn):
public class Main { public static void main(String[] args) { Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(Car.class); enhancer.setCallback(new MethodInterceptor() { @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { System.out.println("before"); Object res = methodProxy.invokeSuper(obj, args); System.out.println("after"); return res; } }); Car car = (Car) enhancer.create(); car.print(); } } class Car { void print() { System.out.println("執(zhí)行原方法"); } }
由于CGLIB并非JDK自帶,所以需要通過Maven引入一個(gè)依賴
<dependency> <groupId>org.sonatype.sisu.inject</groupId> <artifactId>cglib</artifactId> <version>3.1.1</version> </dependency>
輸出結(jié)果:
before
執(zhí)行原方法
after
3、JDK代理與CGLIB代理對(duì)比
1、JDK代理只能實(shí)現(xiàn)接口方法,而CGLIB代理既可以實(shí)現(xiàn)接口方法也可以實(shí)現(xiàn)類中自帶的方法
2、性能上,在JDK1.8,CGLIB3.1.1的環(huán)境上,每次創(chuàng)建一個(gè)代理類并執(zhí)行同樣的方法
當(dāng)執(zhí)行10000次,JDK代理用時(shí)85ms,而CGLIB代理用時(shí)190ms,明顯JDK代理性能更佳;
當(dāng)執(zhí)行1000000(一百萬)次時(shí),兩種代理耗時(shí)幾乎相等;
當(dāng)執(zhí)行10000000次時(shí),CGLIB代理已經(jīng)優(yōu)于JDK代理。
所以在執(zhí)行次數(shù)少時(shí),JDK代理性能更好;反之CGLIB代理性能更好(但是重復(fù)執(zhí)行多于1000000次的任務(wù)幾乎沒有吧
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Spring Boot使用GridFS實(shí)現(xiàn)文件的上傳和下載方式
這篇文章主要介紹了Spring Boot使用GridFS實(shí)現(xiàn)文件的上傳和下載方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10Java連接數(shù)據(jù)庫(kù)實(shí)現(xiàn)方式
文章講述了Java連接MySQL數(shù)據(jù)庫(kù)的詳細(xì)步驟,包括下載和導(dǎo)入JDBC驅(qū)動(dòng)、創(chuàng)建數(shù)據(jù)庫(kù)和表、以及編寫連接和讀取數(shù)據(jù)的代碼2024-11-11RestTemplate使用之如何設(shè)置請(qǐng)求頭、請(qǐng)求體
這篇文章主要介紹了RestTemplate使用之如何設(shè)置請(qǐng)求頭、請(qǐng)求體問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07idea?intellij快速修復(fù)if語(yǔ)句缺少大括號(hào)的問題
這篇文章主要介紹了idea?intellij快速修復(fù)if語(yǔ)句缺少大括號(hào)的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04Java里得到00:00:00格式的時(shí)分秒的Timestamp
Java里如何得到00:00:00格式的時(shí)分秒的Timestamp ,下面是具體的實(shí)現(xiàn)代碼,需要的朋友可以參考下。2009-09-09Spring Boot 快速搭建微服務(wù)框架詳細(xì)教程
SpringBoot是為了簡(jiǎn)化Spring應(yīng)用的創(chuàng)建、運(yùn)行、調(diào)試、部署等而出現(xiàn)的,使用它可以做到專注于Spring應(yīng)用的開發(fā),而無需過多關(guān)注XML的配置。本文重點(diǎn)給大家介紹Spring Boot 快速搭建微服務(wù)框架詳細(xì)教程,需要的的朋友參考下吧2017-09-09Spring實(shí)戰(zhàn)之@Autowire注解用法詳解
這篇文章主要介紹了Spring實(shí)戰(zhàn)之@Autowire注解用法,結(jié)合實(shí)例形式詳細(xì)分析了Spring @Autowire注解具體實(shí)現(xiàn)步驟與相關(guān)使用技巧,需要的朋友可以參考下2019-12-12Spring Boot實(shí)現(xiàn)功能的統(tǒng)一詳解
這篇文章主要介紹了Spring Boot統(tǒng)一功能的處理,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06