电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)

瀏覽:3日期:2022-08-09 10:05:05

好幾天不寫文章,今天來寫一篇,從之前的計劃表上看到還有關于java的動態(tài)代理沒寫,這個技術平常用的少,也不是特別好理解,今天補上這篇,希望能講明白,不至于像我一樣迷茫好久,開始吧

動態(tài)代理分兩部分,動態(tài)和代理,我們先說下代理模式

1、代理模式

Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)

代理模式是常用的設計模式之一,也是開發(fā)中常見的設計模式。

簡單的描述一下,代理模式就是將實現類隔離開,比如你想給你女朋友過個生日,找個明星唱生日歌,你女朋友的偶像是周杰倫,想找周杰倫給她過生日,唱歌,但是你不太能聯系上周杰倫,即使在社交網站上聯系,可能也不太理你,所以你可以聯系周杰倫的經紀人進行溝通,經紀人就是周杰倫的代理。

實現過程:

定義一個唱歌的接口,代表業(yè)務

public interface ISing { void sing();}

周杰倫有唱歌的業(yè)務,并且業(yè)務突出,實現接口

/*** 周杰倫*/public class JayImp implements ISing { @Override public void sing() { System.out.println('say happy birthday to you girl friend'); }}

經紀人接受業(yè)務, 經紀人的構造函數需要和明星綁定

經紀人接收唱歌的業(yè)務,今天可能是周杰倫唱,明天可能經紀人換了明星,比如蔡依林也是可以的

/*** 經紀人*/public class JayProxy implements ISing{ ISing target; /** * 初始化的時候,和明星進行簽約 * @param target */ public JayProxy(ISing target) { this.target = target; } @Override public void sing() { target.sing(); }}

聯系經紀人進行唱歌,周杰倫唱完歌之后,經紀人收錢,very happy

public class MoneyOwner { public static void main(String[] args) { JayImp jay = new JayImp(); // 周杰倫和經紀人進行簽約,這一步可以放在內部實現 JayProxy jayProxy = new JayProxy(jay); jayProxy.sing(); }}

看下執(zhí)行結果,皆大歡喜,你女朋友很開心。

Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)

上面這一套就是代理模式的實現,

但是代理類只能代理一種類,如果為每一個服務都創(chuàng)建一個代理類,有點傻

而且接口如果改變的情況下代理類也需要改變,非常不方便,周杰倫又是拍電影,做綜藝,寫歌,業(yè)務很多

好了,靜態(tài)代理該說的也說了,相信看到這里你應該沒有什么不理解的,下面我們正式開始今天的正餐,動態(tài)代理

2、動態(tài)代理

動態(tài)代理是Java提供的一種代理方式,這個技術的核心點就是在運行期的時候對接口進行增強,生成class 對象,然后加載進虛擬機,說簡單點就是虛擬機幫你創(chuàng)建了一個實現你接口的class

廢話少說,先來實現一個動態(tài)代理

第一步定義接口,上面代碼已經有了 ISing 就不重復定義了

第二步 實現接口,上面代碼也已經實現了 JayImp ,也不重復定義了,這次經紀人多簽約了一個歌手,林俊杰,看下實現

package org.pdool.dynamic; /*** 林俊杰*/public class JJImp implements ISing { @Override public void sing() { System.out.println('I am JJ! happy birthday to you'); }}

第三步,經紀人可以動態(tài)派出簽約歌手,注意經紀人要實現InvocationHandler,這樣才能統(tǒng)一處理所有的方法調用

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy; public class JayAgentFactory implements InvocationHandler { Object target; public JayAgentFactory(Object target) { this.target = target; } // 生成代理類 public ISing CreatProxyedObj() { return (ISing) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object invoke = method.invoke(target, args); logAfter(invoke); return invoke; } public void logAfter(Object invoke) { System.out.println('結果 ' + invoke); System.out.println('收入 ++ '); }}

第四步,接收業(yè)務

package org.pdool.dynamic; import java.lang.reflect.Proxy; public class Aain { public static void main(String[] args) { JayImp jayImp = new JayImp(); ISing subjectProxy=(ISing) Proxy.newProxyInstance(jayImp.getClass().getClassLoader(), jayImp.getClass().getInterfaces(), new JayAgentFactory(jayImp)); subjectProxy.sing(); }}

總結:動態(tài)代理是Java提供的實現方式,需要InvocationHandler 的實現類

1、為什么編輯器可以提示接口的方法?因為強轉編輯器才會能有提示

2、生成的內存class是的默認構造函數是需要InvocationHandler參數

3、創(chuàng)建代理class的核心參數是 類加載器,接口,還有InvocationHandler 子類。

類加載器保證和目標類在同一個加載器內,可以調用,防止不同加載器加載的類之間不能調用

接口就是你要代理的接口

InvocationHandler 子類是轉發(fā)器,將所有的消息進行攔截處理轉發(fā)

3、原理研究

實現看到了,探究下原理,動態(tài)代理的最根本的在于根據接口創(chuàng)建內存class,這一步是怎么實現的,我們跟著源碼瞧一瞧

Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)

1、克隆接口里函數的信息

2、查找或生成指定的代理類,如果緩存中有,則用緩存的,沒有則創(chuàng)建

3、通過反射,拿到代理類的構造函數

4、通過構造函數創(chuàng)建一個代理對象,并關聯InvocationHandler 的對象

/** parameter types of a proxy class constructor */ private static final Class<?>[] constructorParams = { InvocationHandler.class };

看到了流程,我們看下代理class 到底是什么樣子的,

import sun.misc.ProxyGenerator; public class Test { public static void main(String[] args) { //開啟保存代碼class屬性 System.getProperties().put('sun.misc.ProxyGenerator.saveGeneratedFiles', 'true'); ProxyGenerator.generateProxyClass('Xiangcai', JayImp.class.getInterfaces()); }}

執(zhí)行上面的函數,可以看到在項目的路徑下生成Xiangcai.class

Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)

接著看看xiangcai.class 到底有哪些東西,直接拖到編輯器就可以了

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.lang.reflect.UndeclaredThrowableException;import org.pdool.dynamic.ISing; public final class xiangcai extends Proxy implements ISing { private static Method m1; private static Method m2; private static Method m3; private static Method m0; public xiangcai(InvocationHandler var1) throws { super(var1); } public final boolean equals(Object var1) throws { try { return (Boolean)super.h.invoke(this, m1, new Object[]{var1}); } catch (RuntimeException | Error var3) { throw var3; } catch (Throwable var4) { throw new UndeclaredThrowableException(var4); } } public final String toString() throws { try { return (String)super.h.invoke(this, m2, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } }//注意看這里!!!其他的都不重要 public final void sing() throws { try { super.h.invoke(this, m3, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } public final int hashCode() throws { try { return (Integer)super.h.invoke(this, m0, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } static { try { m1 = Class.forName('java.lang.Object').getMethod('equals', Class.forName('java.lang.Object')); m2 = Class.forName('java.lang.Object').getMethod('toString'); m3 = Class.forName('org.pdool.dynamic.ISing').getMethod('sing'); m0 = Class.forName('java.lang.Object').getMethod('hashCode'); } catch (NoSuchMethodException var2) { throw new NoSuchMethodError(var2.getMessage()); } catch (ClassNotFoundException var3) { throw new NoClassDefFoundError(var3.getMessage()); } }}

可以看到實現了sing的接口,并且調用了invokehandler的方法invoke.好了,真相大白了,你明白了嗎?

有人會說,道理我都懂,可是不會用啊,但是沒看到好的應用場景,所以有段時間是沒掌握這些的,下面我們就具體一下應用場景

4、應用

在切面編程(AOP)中,需要攔截特定的方法,通常,會選擇動態(tài)代理方式。看個具體的例子 spring-data-jpa 的實現

具體的使用:

spring中訪問數據庫的使用

import com.tao.springboot.hibernate.entity.Customer;import org.springframework.data.jpa.repository.JpaRepository; public interface CustomerRepository extends JpaRepository<Customer, Long> { }

只要實現上面的接口就可以直接操作數據庫,是不是很簡單?

有幾個問題,你稍微思考下:

1、兩個泛型什么意思?

2、數據庫連接在哪?是怎么注入的?

3、只實現接口是怎么操作數據庫的?

第一個問題答案:

Customer 為表對象對應的entity實體。

Long 是表的主鍵類型,

第二個答案:

數據庫連接是在spring啟動的時候自動注入到spring 容器中的,在JpaRepository 的實現類自動注入的

第三個答案:

所有的的接口在spring啟動的時候會生成代理類,目標類target就是實現類SimpleJpaRepository

看下類圖

Java 動態(tài)代理你真的懂了嗎(動態(tài)和代理)

看下JpaRepository的定義,都是一些常用方法

public interface JpaRepository<T, ID> extends PagingAndSortingRepository<T, ID>, QueryByExampleExecutor<T> { List<T> findAll(); List<T> findAll(Sort var1); List<T> findAllById(Iterable<ID> var1); <S extends T> List<S> saveAll(Iterable<S> var1); void flush(); <S extends T> S saveAndFlush(S var1); void deleteInBatch(Iterable<T> var1); void deleteAllInBatch(); T getOne(ID var1); <S extends T> List<S> findAll(Example<S> var1); <S extends T> List<S> findAll(Example<S> var1, Sort var2);}

看下SimpleJpaRepository 的定義:

public class SimpleJpaRepository<T, ID> implements JpaRepositoryImplementation<T, ID> { private static final String ID_MUST_NOT_BE_NULL = 'The given id must not be null!'; private final JpaEntityInformation<T, ?> entityInformation; private final EntityManager em;//看這里!!! private final PersistenceProvider provider; @Nullable private CrudMethodMetadata metadata; //具體的實現方法 @Transactional public void delete(T entity) { Assert.notNull(entity, 'The entity must not be null!'); this.em.remove(this.em.contains(entity) ? entity : this.em.merge(entity)); }

類似下面的代碼調用:

Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(), JpaRepository.class.getInterfaces(), new SimpleJpaRepository(());

注:只是表達一下意思,具體的實現應該不是這樣的

5、總結

所有的事情都解開了,下面進行總結下:

1、靜態(tài)代理是代理模式的實現,是針對某一個具體的接口的實現

2、動態(tài)代理的是jdk提供的一種方式,必須要接口,還有其他的實現方式cglib,javassit 等等

3、動態(tài)代理是在運行的時候生成class 文件然后自動加載的class

4、動態(tài)代理是基于反射調用的技術

5、動態(tài)代理會生成class 到 metaspace

6、多應用在框架中

7、解密了spring data jpa的實現

到此這篇關于Java 動態(tài)代理都不懂怎么裝逼?的文章就介紹到這了,更多相關Java 動態(tài)代理內容請搜索好吧啦網以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
主站蜘蛛池模板: 铣刨料沥青破碎机-沥青再生料设备-RAP热再生混合料破碎筛分设备 -江苏锡宝重工 | C形臂_动态平板DR_动态平板胃肠机生产厂家制造商-普爱医疗 | 档案密集架_电动密集架_移动密集架_辽宁档案密集架-盛隆柜业厂家现货批发销售价格公道 | 上海恒驭仪器有限公司-实验室平板硫化机-小型平板硫化机-全自动平板硫化机 | 泵阀展|阀门展|水泵展|流体机械展 -2025上海国际泵管阀展览会flowtech china | 烽火安全网_加密软件、神盾软件官网 | 干粉砂浆设备_干混砂浆生产线_腻子粉加工设备_石膏抹灰砂浆生产成套设备厂家_干粉混合设备_砂子烘干机--郑州铭将机械设备有限公司 | 除尘器布袋骨架,除尘器滤袋,除尘器骨架,电磁脉冲阀膜片,卸灰阀,螺旋输送机-泊头市天润环保机械设备有限公司 | 云南丰泰挖掘机修理厂-挖掘机维修,翻新,再制造的大型企业-云南丰泰工程机械维修有限公司 | Type-c防水母座|贴片母座|耳机接口|Type-c插座-深圳市步步精科技有限公司 | 【化妆品备案】进口化妆品备案流程-深圳美尚美化妆品有限公司 | 电表箱-浙江迈峰电力设备有限公司-电表箱专业制造商 | 沈阳缠绕膜价格_沈阳拉伸膜厂家_沈阳缠绕膜厂家直销 | 雷冲击高压发生器-水内冷直流高压发生器-串联谐振分压器-武汉特高压电力科技有限公司 | WF2户外三防照明配电箱-BXD8050防爆防腐配电箱-浙江沃川防爆电气有限公司 | 吹塑加工_大型吹塑加工_滚塑代加工-莱力奇吹塑加工有限公司 | 304不锈钢无缝管_不锈钢管厂家 - 隆达钢业集团有限公司 | Copeland/谷轮压缩机,谷轮半封闭压缩机,谷轮涡旋压缩机,型号规格,技术参数,尺寸图片,价格经销商 CTP磁天平|小电容测量仪|阴阳极极化_双液系沸点测定仪|dsj电渗实验装置-南京桑力电子设备厂 | 耐酸碱泵-自吸耐酸碱泵型号「品牌厂家」立式耐酸碱泵价格-昆山国宝过滤机有限公司首页 | 对夹式止回阀_对夹式蝶形止回阀_对夹式软密封止回阀_超薄型止回阀_不锈钢底阀-温州上炬阀门科技有限公司 | 许昌奥仕达自动化设备有限公司 | 无锡市珂妮日用化妆品有限公司|珂妮日化官网|洗手液厂家 | 德州网站开发定制-小程序开发制作-APP软件开发-「两山开发」 | 永嘉县奥阳陶瓷阀门有限公司 | 避光流动池-带盖荧光比色皿-生化流动比色皿-宜兴市晶科光学仪器 东莞爱加真空科技有限公司-进口真空镀膜机|真空镀膜设备|Polycold维修厂家 | 安平县鑫川金属丝网制品有限公司,防风抑尘网,单峰防风抑尘,不锈钢防风抑尘网,铝板防风抑尘网,镀铝锌防风抑尘网 | 螺杆泵_中成泵业| 锂电池生产厂家-电动自行车航模无人机锂电池定制-世豹新能源 | 铸铝门厂家,别墅大门庭院大门,别墅铸铝门铜门[十大品牌厂家]军强门业 | 折弯机-刨槽机-数控折弯机-数控刨槽机-数控折弯机厂家-深圳豐科机械有限公司 | 欧盟ce检测认证_reach检测报告_第三方检测中心-深圳市威腾检验技术有限公司 | 玻璃钢型材-玻璃钢风管-玻璃钢管道,生产厂家-[江苏欧升玻璃钢制造有限公司] | 深圳美安可自动化设备有限公司,喷码机,定制喷码机,二维码喷码机,深圳喷码机,纸箱喷码机,东莞喷码机 UV喷码机,日期喷码机,鸡蛋喷码机,管芯喷码机,管内壁喷码机,喷码机厂家 | FAG轴承,苏州FAG轴承,德国FAG轴承-恩梯必传动设备(苏州)有限公司 | 上海防爆真空干燥箱-上海防爆冷库-上海防爆冷柜?-上海浦下防爆设备厂家? | 硅胶布|电磁炉垫片|特氟龙胶带-江苏浩天复合材料有限公司 | 塑钢件_塑钢门窗配件_塑钢配件厂家-文安县启泰金属制品有限公司 深圳南财多媒体有限公司介绍 | 青州搬家公司电话_青州搬家公司哪家好「鸿喜」青州搬家 | 深圳工程师职称评定条件及流程_深圳职称评审_职称评审-职称网 | 深圳宣传片制作-企业宣传视频制作-产品视频拍摄-产品动画制作-短视频拍摄制作公司 | 液压油缸-液压缸厂家价格,液压站系统-山东国立液压制造有限公司 液压油缸生产厂家-山东液压站-济南捷兴液压机电设备有限公司 |