溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java動(dòng)態(tài)代理常用方式有哪些

發(fā)布時(shí)間:2021-11-22 17:08:13 來(lái)源:億速云 閱讀:142 作者:iii 欄目:開(kāi)發(fā)技術(shù)

本篇內(nèi)容介紹了“Java動(dòng)態(tài)代理常用方式有哪些”的有關(guān)知識(shí),在實(shí)際案例的操作過(guò)程中,不少人都會(huì)遇到這樣的困境,接下來(lái)就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

    什么是動(dòng)態(tài)代理?

    動(dòng)態(tài)代理就是,在程序運(yùn)行期,創(chuàng)建目標(biāo)對(duì)象的代理對(duì)象,并對(duì)目標(biāo)對(duì)象中的方法進(jìn)行功能性增強(qiáng)的一種技術(shù)。在生成代理對(duì)象的過(guò)程中,目標(biāo)對(duì)象不變,代理對(duì)象中的方法是目標(biāo)對(duì)象方法的增強(qiáng)方法??梢岳斫鉃檫\(yùn)行期間,對(duì)象中方法的動(dòng)態(tài)攔截,在攔截方法的前后執(zhí)行功能操作(也可以對(duì)原方法的參數(shù)進(jìn)行操作)。

    代理類在程序運(yùn)行期間,創(chuàng)建的代理對(duì)象稱之為動(dòng)態(tài)代理對(duì)象。這種情況下,創(chuàng)建的代理對(duì)象,并不是事先在Java代碼中定義好的。而是在運(yùn)行期間,根據(jù)我們?cè)趧?dòng)態(tài)代理對(duì)象中的“指示”,動(dòng)態(tài)生成的。也就是說(shuō),你想獲取哪個(gè)對(duì)象的代理,動(dòng)態(tài)代理就會(huì)為你動(dòng)態(tài)的生成這個(gè)對(duì)象的代理對(duì)象。動(dòng)態(tài)代理可以對(duì)被代理對(duì)象的方法進(jìn)行功能增強(qiáng)。有了動(dòng)態(tài)代理的技術(shù),那么就可以在不修改方法源碼的情況下,增強(qiáng)被代理對(duì)象的方法的功能,在方法執(zhí)行前后做任何你想做的事情。

    特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載

    作用:不修改源碼的基礎(chǔ)上對(duì)方法增強(qiáng)

    動(dòng)態(tài)代理的常用兩種方式:

    1.基于接口的動(dòng)態(tài)代理

    提供者:JDK

    使用JDK官方的Proxy類創(chuàng)建代理對(duì)象

    注意:代理的目標(biāo)對(duì)象必須實(shí)現(xiàn)接口(至少一個(gè))

    2.基于類的動(dòng)態(tài)代理

    提供者:第三方 CGLib

    使用CGLib的Enhancer類創(chuàng)建代理對(duì)象

    注意:被代理類不能用 final 修飾的類(最終類)。如果報(bào) asmxxxx 異常,需要導(dǎo)入 asm.jar包

    //JDK動(dòng)態(tài)代理(基于接口的動(dòng)態(tài)代理)
    Proxy.newProxyInstance(三個(gè)參數(shù));
    	ClassLoader:類加載器
    		它是用于加載代理對(duì)象字節(jié)碼的。和被代理對(duì)象使用相同的類加載器。(固定寫(xiě)法)
    	Class[]:字節(jié)碼數(shù)組
    		它是用于讓代理對(duì)象和被代理對(duì)象有相同方法。(固定寫(xiě)法)
    	InvocationHandler:用于提供增強(qiáng)的代碼
    		它是讓我們寫(xiě)如何代理。我們一般都是寫(xiě)一個(gè)該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須
    		InvocationHandler該接口的實(shí)現(xiàn)類是誰(shuí)用誰(shuí)寫(xiě),此時(shí)我們用就需要我們自己寫(xiě)

    此處以一個(gè)演員的例子為例:

    在很久以前,演員和劇組都是直接見(jiàn)面聯(lián)系的。沒(méi)有中間人環(huán)節(jié)。

    而隨著時(shí)間的推移,產(chǎn)生了一個(gè)新興職業(yè):經(jīng)紀(jì)人(中間人),這個(gè)時(shí)候劇組再想找演員就需要通過(guò)經(jīng)紀(jì)人來(lái)找了。下面我們就用代碼演示出來(lái)。

    package com.haust.service;
    public interface IActor {
    	/**
    	* 基本演出
    	* @param money
    	*/
    	public void basicAct(float money);
    	/**
    	* 危險(xiǎn)演出
    	* @param money
    	*/
    	public void dangerAct(float money);
    }
    package com.haust.serviceImpl;
    import com.haust.service.IActor;
    public class Actor implements IActor {
    	/**
    	* 一個(gè)演員
    	*/
    	//實(shí)現(xiàn)了接口,就表示具有接口中的方法實(shí)現(xiàn)。即:符合經(jīng)紀(jì)公司的要求
    	@Override
    	public void basicAct(float money) {
    		System.out.println("拿到錢(qián),開(kāi)始基本的表演:"+money);
    	}
    	@Override
    	public void dangerAct(float money) {
    		System.out.println("拿到錢(qián),開(kāi)始危險(xiǎn)的表演:"+money);
    	}
    }
    package com.haust.test;
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    import com.haust.service.IActor;
    import com.haust.serviceImpl.Actor;
    public class Client {
    	public static void main(String[] args) {
    		//一個(gè)劇組找演員:
    		final Actor actor = new Actor();//被代理的類
    		/**
    		* 代理:
    		* 間接。
    		* 獲取代理對(duì)象:
    		* 要求:
    		* 被代理類最少實(shí)現(xiàn)一個(gè)接口
    		* 創(chuàng)建的方式
    		* Proxy.newProxyInstance(三個(gè)參數(shù))
    		* 參數(shù)含義:
    		* ClassLoader:和被代理對(duì)象使用相同的類加載器。
    		* Interfaces:和被代理對(duì)象具有相同的行為。實(shí)現(xiàn)相同的接口。
    		* InvocationHandler:如何代理。
    		* 
    		*/
    		//(IActor)Proxy.newProxyInstance,這里強(qiáng)制轉(zhuǎn)換必須是接口類型
    		IActor proxyActor = (IActor)Proxy.newProxyInstance(actor.getClass().getClassLoader(),
    				 actor.getClass().getInterfaces(),
    				 new InvocationHandler() {
    			 /**
    			 * 執(zhí)行被代理對(duì)象的任何方法,都會(huì)經(jīng)過(guò)該方法。
    			 * 此方法有攔截的功能。
    			 * 
    			 * 參數(shù):
    			 * proxy:代理對(duì)象的引用。不一定每次都用得到
    			 * method:當(dāng)前執(zhí)行的方法對(duì)象
    			 * args:執(zhí)行方法所需的參數(shù)
    			 * 返回值:
    			 * 當(dāng)前執(zhí)行被代理對(duì)象方法的返回值
    			 */
    					@Override
    					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    						String name = method.getName();
    						Float money = (Float) args[0];//執(zhí)行的方法只有一個(gè)參數(shù)
    						Object rtValue = null;
    						//每個(gè)經(jīng)紀(jì)公司對(duì)不同演出收費(fèi)不一樣,此處開(kāi)始判斷
    						if("basicAct".equals(name)){
    						//基本演出,沒(méi)有 2000 不演
    						if(money > 2000){
    						//看上去劇組是給了 8000,實(shí)際到演員手里只有 4000
    						//這就是我們沒(méi)有修改原來(lái) basicAct 方法源碼,對(duì)方法進(jìn)行了增強(qiáng)
    						rtValue = method.invoke(actor, money/2);
    						} }
    						if("dangerAct".equals(name)){
    						//危險(xiǎn)演出,沒(méi)有 5000 不演
    						if(money > 5000){
    						//看上去劇組是給了 50000,實(shí)際到演員手里只有 25000
    						//這就是我們沒(méi)有修改原來(lái) dangerAct 方法源碼,對(duì)方法進(jìn)行了增強(qiáng)
    						rtValue = method.invoke(actor, money/2);
    						} }
    						return rtValue;
    					}
    				});
    		//沒(méi)有經(jīng)紀(jì)公司的時(shí)候,直接找演員。
    		// actor.basicAct(1000f);
    		// actor.dangerAct(5000f);
    		//劇組無(wú)法直接聯(lián)系演員,而是由經(jīng)紀(jì)公司找的演員
    		proxyActor.basicAct(2000f);//價(jià)格低于2000不演
    		proxyActor.dangerAct(50000f);
    		
    	}
    }

    總結(jié):

    首先需要?jiǎng)?chuàng)建一個(gè)interface然后一個(gè)class實(shí)現(xiàn)這個(gè)interface,然后對(duì)這個(gè)class進(jìn)行代理,這個(gè)class必須實(shí)現(xiàn)至少一個(gè)接口

    基于子類的動(dòng)態(tài)代理

    設(shè)計(jì)的類:Enhancer
    提供者:第三方cglib庫(kù)
    如何創(chuàng)建代理對(duì)象:
    	使用Enhancer類中的create方法
    創(chuàng)建代理對(duì)象的要求:
    	被代理對(duì)象不是最終類(最終類沒(méi)有子類)
    create方法的參數(shù):
    	Class方法的參數(shù):
    		Class:字節(jié)碼
    			它是用于指定被代理對(duì)象的字節(jié)碼
    		callback:用于提供增強(qiáng)的代碼
    			它是讓我們寫(xiě)如何代理。我們一般都是些一個(gè)該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的。此接口的實(shí)現(xiàn)類都是誰(shuí)用誰(shuí)寫(xiě)。
    			我們一般寫(xiě)的都是該接口的子接口實(shí)現(xiàn)類:MethodInterceptor
    //CGLib動(dòng)態(tài)代理(基于子類的動(dòng)態(tài)代理)
    Enhancer.create(兩個(gè)參數(shù));

    代碼如下:

    package com.haust.serviceImpl;
    
    public class Actor{//沒(méi)有實(shí)現(xiàn)任何接口
    	/**
    	* 一個(gè)演員
    	*/
    	public void basicAct(float money) {
    		System.out.println("拿到錢(qián),開(kāi)始基本的表演:"+money);
    	}
    	public void dangerAct(float money) {
    		System.out.println("拿到錢(qián),開(kāi)始危險(xiǎn)的表演:"+money);
    	}
    }
    package com.haust.test;
    import java.lang.reflect.Method;
    import org.springframework.cglib.proxy.Enhancer;
    import org.springframework.cglib.proxy.MethodInterceptor;
    import org.springframework.cglib.proxy.MethodProxy;
    import com.haust.serviceImpl.Actor;
    public class test {
    	public static void main(String[] args) {
    		 Actor actor = new Actor();//需要?jiǎng)?chuàng)建此被代理的對(duì)象
    		 /**
    		 * 基于子類的動(dòng)態(tài)代理
    		 * 要求:
    		 * 被代理對(duì)象不能是最終類
    		 * 用到的類:
    		 * Enhancer
    		 * 用到的方法:
    		 * create(Class, Callback)
    		 * 方法的參數(shù):
    		 * Class:被代理對(duì)象的字節(jié)碼
    		 * Callback:如何代理
    		 * @param args
    		 */
    		 //此時(shí)強(qiáng)轉(zhuǎn)的類的類型就是被代理類的類型
    		 Actor cglibActor = (Actor)Enhancer.create(actor.getClass(),new MethodInterceptor() {
    			@Override
    			public Object intercept(Object proxy, Method method, Object[] args, MethodProxy  methodProxy) throws Throwable {
    				/**
    				* 執(zhí)行被代理對(duì)象的任何方法,都會(huì)經(jīng)過(guò)該方法。在此方法內(nèi)部就可以對(duì)被代理對(duì)象的任何
    				方法進(jìn)行增強(qiáng)。
    				* 
    				* 參數(shù):
    				* 前三個(gè)和基于接口的動(dòng)態(tài)代理是一樣的。
    				* MethodProxy:當(dāng)前執(zhí)行方法的代理對(duì)象。
    				* 返回值:
    				* 當(dāng)前執(zhí)行方法的返回值
    				*/
    				String name = method.getName();
    				Float money = (Float) args[0];
    				Object rtValue = null;
    				if("basicAct".equals(name)){
    				//基本演出
    				if(money > 2000){
    				rtValue = method.invoke(actor, money/2);
    				} }
    				if("dangerAct".equals(name)){
    					//危險(xiǎn)演出
    					if(money > 5000){
    					rtValue = method.invoke(actor, money/2);
    					} 
    				}
    				return rtValue;
    			}
    		});
    		 cglibActor.basicAct(10000);
    		 cglibActor.dangerAct(100000);
    	}
    }

    “Java動(dòng)態(tài)代理常用方式有哪些”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

    向AI問(wèn)一下細(xì)節(jié)

    免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

    AI