詳細分析java 動態代理
1、動態代理的特點:
字節碼隨用隨創建,隨用隨加載
2、作用:
不修改源碼的基礎上對源碼進行加強
3、分類:
(1)基于接口的動態代理:
涉及到的類:Proxy,由JDK官方提供,使用Proxy類中的newProxyInstance方法創建對象。創建代理對象時要求被代理對象至少實現一個接口,否則無法使用
參數:
ClassLoader:類加載器,他是用于加載對象字節碼的,和被代理對象使用相同的類加載器,為固定寫法 class[]:字節碼數組,他是用于讓代理對象和被代理對象具有相同的方法,也是固定寫法 InvocationHandler:用戶提供增強的代碼 ,他是讓我們寫如何代理。我們一般都是寫一個該接口的實現類,通常情況下都是匿名內部類,但不是必須的,此接口的實現類都是誰用誰寫示例: 創建 Producter接口和實體類
package com.mingqi.proxy;/** * 對生產廠家要求的接口 */public interface IProducer { /** * 銷售 * @param money */ public void SaleProduct(float money); /** * 售后 * @param money */ public void AfterService(float money);}
package com.mingqi.proxy;public class Producer implements IProducer { public void SaleProduct(float money) { System.out.println('銷售產品,并拿到錢:'+money); } public void AfterService(float money) { System.out.println('提供售后服務,并拿到錢:'+money); }}
測試方法:
public static void main(String[] args) { /* 1、動態代理 特點:字節碼隨用隨創建,隨用隨加載 作用:不修改源碼的基礎上對源碼進行加強 分類:基于接口的動態代理 涉及的類: Proxy 提供者:JDK官方 如何創建代理對象: 使用Proxy類中的newProxyInstance方法 創建代理對象的要求: 被代理對象至少實現一個接口,如果沒有則不能使用 newProxyInstance的方法參數: ClassLoader:類加載器 他是用于加載代理對象字節碼的,和被代理對象使用相同的類加載器,固定寫法 class[] :字節碼數組 InvocationHandler 用于提供增強的代碼 他是讓我們寫如何代理,我們一般都是寫一個接口的實現類,通常情況下都是匿名內部類,但不是必須的,此接口的實現類都是誰用誰寫*/ final Producer producer=new Producer(); IProdurcer proxyProducer= (IProdurcer)Proxy.newProxyInstance(producer.getClass().getClassLoader(), producer.getClass().getInterfaces(), new InvocationHandler() { /** * 作用:執行被代理對象的任何接口方法都會經過該方法 * 方法參數的含義 * @param proxy 代理對象的引用 * @param method 當前執行的方法 * @param args 當前執行方法所需的參數 * @return 被代理對象有相同的返回值 * @throws Throwable */ public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //提供增強的代碼: Object returnValue=null; //1、獲取方法執行的參數 Float money=(Float) args[0]; //判斷當前方法是否是銷售 if('SaleProduct'.equals(method.getName())) { returnValue=method.invoke(producer,money*0.8f); } return returnValue; } }); proxyProducer.SaleProduct(10000f); }
(2) 基于子類的動態代理
涉及到的類:Enhancer,由第三方cglib提供,使用Enhancer類中的create方法創建對象。創建代理對象的類不能是最終類,否則無法使用
參數:
Class:字節碼,他是用于指定被代理對象的字節碼,為固定寫法 Callback:用戶提供增強的代碼 ,他是讓我們寫如何代理。我們一般都是寫一個該接口的實現類,通常情況下都是匿名內部類,但不是必須的,此接口的實現類都是誰用誰寫,我們一般寫的都是該接口的子接口實現類MethodInterceptor示例: 創建 Product接口和實體類
package com.mingqi.cglib;/** * 一個生產者 */public class Product { /** * 銷售 * @param money */ public void saleProduct(float money){ System.out.println('銷售產品,并拿到錢:'+money); } /** * 售后 * @param money */ public void afterService(float money){ System.out.println('提供售后服務,并拿到錢:'+money); }}
測試類及方法:
package com.mingqi.cglib;import net.sf.cglib.proxy.Enhancer;import net.sf.cglib.proxy.MethodInterceptor;import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;public class Client { public static void main(String[] args) { final Product product=new Product(); Product cglibproduct= (Product) Enhancer.create(product.getClass(), new MethodInterceptor() { public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {Object returnValue=null;//1、獲取方法執行的參數Float money=(Float)objects[0];//判斷當前方法是不是銷售if('SaleProduct'.equals(method.getName())) { returnValue = method.invoke(product, money*0.8f);}return returnValue; } }); cglibproduct.SaleProduct(1000f); }}
以上就是創建動態代理對象的兩種類型,以后要經常練習使用,讓這種思想能給我們工作中帶來方便。
到此這篇關于詳細分析java 動態代理的文章就介紹到這了,更多相關java 動態代理內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!
相關文章: