" />

欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

帶你重新認(rèn)識Java動(dòng)態(tài)代理

 更新時(shí)間:2021年11月22日 15:45:46   作者:浪里小白龍nbw  
這篇文章主要為大家介紹了Java的動(dòng)態(tài)代理,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助

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

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

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

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

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

正常類創(chuàng)建對象的過程:

在這里插入圖片描述

動(dòng)態(tài)代理創(chuàng)建代理對象的過程:

在這里插入圖片描述

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

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

提供者:JDK

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

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

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

提供者:第三方 CGLib

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

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

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

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

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

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

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("拿到錢,開始基本的表演:"+money);
	}
	@Override
	public void dangerAct(float money) {
		System.out.println("拿到錢,開始危險(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();//被代理的類
		/**
		* 代理:
		* 間接。
		* 獲取代理對象:
		* 要求:
		* 被代理類最少實(shí)現(xiàn)一個(gè)接口
		* 創(chuàng)建的方式
		* Proxy.newProxyInstance(三個(gè)參數(shù))
		* 參數(shù)含義:
		* ClassLoader:和被代理對象使用相同的類加載器。
		* Interfaces:和被代理對象具有相同的行為。實(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í)行被代理對象的任何方法,都會(huì)經(jīng)過該方法。
			 * 此方法有攔截的功能。
			 * 
			 * 參數(shù):
			 * proxy:代理對象的引用。不一定每次都用得到
			 * method:當(dāng)前執(zhí)行的方法對象
			 * args:執(zhí)行方法所需的參數(shù)
			 * 返回值:
			 * 當(dāng)前執(zhí)行被代理對象方法的返回值
			 */
					@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ì)公司對不同演出收費(fèi)不一樣,此處開始判斷
						if("basicAct".equals(name)){
						//基本演出,沒有 2000 不演
						if(money > 2000){
						//看上去劇組是給了 8000,實(shí)際到演員手里只有 4000
						//這就是我們沒有修改原來 basicAct 方法源碼,對方法進(jìn)行了增強(qiáng)
						rtValue = method.invoke(actor, money/2);
						} }
						if("dangerAct".equals(name)){
						//危險(xiǎn)演出,沒有 5000 不演
						if(money > 5000){
						//看上去劇組是給了 50000,實(shí)際到演員手里只有 25000
						//這就是我們沒有修改原來 dangerAct 方法源碼,對方法進(jìn)行了增強(qiáng)
						rtValue = method.invoke(actor, money/2);
						} }
						return rtValue;
					}
				});
		//沒有經(jīng)紀(jì)公司的時(shí)候,直接找演員。
		// actor.basicAct(1000f);
		// actor.dangerAct(5000f);
		//劇組無法直接聯(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,然后對這個(gè)class進(jìn)行代理,這個(gè)class必須實(shí)現(xiàn)至少一個(gè)接口

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

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

代碼如下:

package com.haust.serviceImpl;

public class Actor{//沒有實(shí)現(xiàn)任何接口
	/**
	* 一個(gè)演員
	*/
	public void basicAct(float money) {
		System.out.println("拿到錢,開始基本的表演:"+money);
	}
	public void dangerAct(float money) {
		System.out.println("拿到錢,開始危險(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)建此被代理的對象
		 /**
		 * 基于子類的動(dòng)態(tài)代理
		 * 要求:
		 * 被代理對象不能是最終類
		 * 用到的類:
		 * Enhancer
		 * 用到的方法:
		 * create(Class, Callback)
		 * 方法的參數(shù):
		 * Class:被代理對象的字節(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í)行被代理對象的任何方法,都會(huì)經(jīng)過該方法。在此方法內(nèi)部就可以對被代理對象的任何
				方法進(jìn)行增強(qiáng)。
				* 
				* 參數(shù):
				* 前三個(gè)和基于接口的動(dòng)態(tài)代理是一樣的。
				* MethodProxy:當(dāng)前執(zhí)行方法的代理對象。
				* 返回值:
				* 當(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);
	}
}

總結(jié):

無論哪種代理方式,都需要?jiǎng)?chuàng)建一個(gè)被代理的類(實(shí)例)。

不管是基于接口的代理,還是基于子類的代理,均攔截被代理對象的所有方法,然后我們可以對這些方法進(jìn)行增強(qiáng)或者其他一些操作。

本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!

相關(guān)文章

  • Flowable歷史查詢實(shí)例分析

    Flowable歷史查詢實(shí)例分析

    這篇文章主要介紹了Flowable歷史查詢實(shí)例分析,歷史是記錄流程執(zhí)行過程中發(fā)生的事情,并將其永久存儲(chǔ)的組件,與運(yùn)行時(shí)數(shù)據(jù)不同,歷史數(shù)據(jù)在流程實(shí)例完成以后仍保存在數(shù)據(jù)庫中,下面我們來深入了解
    2023-10-10
  • Spring組件開發(fā)模式支持SPEL表達(dá)式

    Spring組件開發(fā)模式支持SPEL表達(dá)式

    今天小編就為大家分享一篇關(guān)于Spring組件開發(fā)模式支持SPEL表達(dá)式,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧
    2018-12-12
  • 基于SPRINGBOOT配置文件占位符過程解析

    基于SPRINGBOOT配置文件占位符過程解析

    這篇文章主要介紹了基于SPRINGBOOT配置文件占位符過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • Spring如何基于注解顯式實(shí)現(xiàn)自動(dòng)裝配

    Spring如何基于注解顯式實(shí)現(xiàn)自動(dòng)裝配

    這篇文章主要介紹了Spring如何基于注解顯式實(shí)現(xiàn)自動(dòng)裝配,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-08-08
  • springboot實(shí)現(xiàn)請求參數(shù)驗(yàn)證的多種方法

    springboot實(shí)現(xiàn)請求參數(shù)驗(yàn)證的多種方法

    在日常開發(fā)中,我們少不了需要對前端的請求參數(shù)的驗(yàn)證,Spring提供了多種方法來實(shí)現(xiàn)請求參數(shù)的驗(yàn)證,文中通過代碼示例給大家講解的非常詳細(xì),我們一起了解一下吧
    2023-11-11
  • 深入理解Java設(shè)計(jì)模式之單例模式

    深入理解Java設(shè)計(jì)模式之單例模式

    這篇文章主要介紹了JAVA設(shè)計(jì)模式之單例模式的的相關(guān)資料,文中示例代碼非常詳細(xì),供大家參考和學(xué)習(xí),感興趣的朋友可以了解下
    2021-11-11
  • Springboot單元測試編寫實(shí)踐

    Springboot單元測試編寫實(shí)踐

    在日常的開發(fā)過程中,為了提高代碼的可靠性和健壯性,同時(shí)也是檢測代碼的質(zhì)量,減少測試環(huán)節(jié)的問題,會(huì)對完成的業(yè)務(wù)功能代碼編寫單元測試,在本文中,將分享一些單元測試的實(shí)踐和心得,需要的朋友可以參考下
    2023-11-11
  • java并發(fā)學(xué)習(xí)-CountDownLatch實(shí)現(xiàn)原理全面講解

    java并發(fā)學(xué)習(xí)-CountDownLatch實(shí)現(xiàn)原理全面講解

    這篇文章主要介紹了java并發(fā)學(xué)習(xí)-CountDownLatch實(shí)現(xiàn)原理全面講解,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • 詳解mybatis批量插入10萬條數(shù)據(jù)的優(yōu)化過程

    詳解mybatis批量插入10萬條數(shù)據(jù)的優(yōu)化過程

    這篇文章主要介紹了詳解mybatis批量插入10萬條數(shù)據(jù)的優(yōu)化過程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • Kafka日志清理實(shí)現(xiàn)詳細(xì)過程講解

    Kafka日志清理實(shí)現(xiàn)詳細(xì)過程講解

    這篇文章主要為大家介紹了Kafka日志清理實(shí)現(xiàn)詳細(xì)過程講解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05

最新評論