代理的分析
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
http://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.learn</groupId><artifactId>day03_learn_02proxy</artifactId><version>1.0-SNAPSHOT</version><packaging>jar</packaging><dependencies><dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>2.1_3</version></dependency></dependencies>
</project>
?
package com.learn.cglib;import net.sf.cglib.proxy.Enhancer; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;/*** 模擬一個消費(fèi)者*/ public class Client {public static void main(String[] args) {final Producer producer = new Producer();/*** 動態(tài)代理:* 特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載* 作用:不修改源碼的基礎(chǔ)上對方法增強(qiáng)* 分類:* 基于接口的動態(tài)代理* 基于子類的動態(tài)代理* 基于子類的動態(tài)代理:* 涉及的類:Enhancer* 提供者:第三方cglib庫* 如何創(chuàng)建代理對象:* 使用Enhancer類中的create方法* 創(chuàng)建代理對象的要求:* 被代理類不能是最終類* create方法的參數(shù):* Class:字節(jié)碼* 它是用于指定被代理對象的字節(jié)碼。** Callback:用于提供增強(qiáng)的代碼* 它是讓我們寫如何代理。我們一般都是些一個該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的。* 此接口的實(shí)現(xiàn)類都是誰用誰寫。* 我們一般寫的都是該接口的子接口實(shí)現(xiàn)類:MethodInterceptor*/Producer cglibProducer = (Producer)Enhancer.create(producer.getClass(), new MethodInterceptor() {/*** 執(zhí)行北地阿里對象的任何方法都會經(jīng)過該方法* @param proxy* @param method* @param args* 以上三個參數(shù)和基于接口的動態(tài)代理中invoke方法的參數(shù)是一樣的* @param methodProxy :當(dāng)前執(zhí)行方法的代理對象* @return* @throws Throwable*/public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {//提供增強(qiáng)的代碼Object returnValue = null;//1.獲取方法執(zhí)行的參數(shù)Float money = (Float)args[0];//2.判斷當(dāng)前方法是不是銷售if("saleProduct".equals(method.getName())) {returnValue = method.invoke(producer, money*0.8f);}return returnValue;}});cglibProducer.saleProduct(12000f);} } package com.learn.cglib;/*** 一個生產(chǎn)者*/ public class Producer {/*** 銷售* @param money*/public void saleProduct(float money){System.out.println("銷售產(chǎn)品,并拿到錢:"+money);}/*** 售后* @param money*/public void afterService(float money){System.out.println("提供售后服務(wù),并拿到錢:"+money);} } package com.learn.proxy;import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy;/*** 模擬一個消費(fèi)者*/ public class Client {public static void main(String[] args) {final Producer producer = new Producer();/*** 動態(tài)代理:* 特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載* 作用:不修改源碼的基礎(chǔ)上對方法增強(qiáng)* 分類:* 基于接口的動態(tài)代理* 基于子類的動態(tài)代理* 基于接口的動態(tài)代理:* 涉及的類:Proxy* 提供者:JDK官方* 如何創(chuàng)建代理對象:* 使用Proxy類中的newProxyInstance方法* 創(chuàng)建代理對象的要求:* 被代理類最少實(shí)現(xiàn)一個接口,如果沒有則不能使用* newProxyInstance方法的參數(shù):* ClassLoader:類加載器* 它是用于加載代理對象字節(jié)碼的。和被代理對象使用相同的類加載器。固定寫法。* Class[]:字節(jié)碼數(shù)組* 它是用于讓代理對象和被代理對象有相同方法。固定寫法。* InvocationHandler:用于提供增強(qiáng)的代碼* 它是讓我們寫如何代理。我們一般都是些一個該接口的實(shí)現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的。* 此接口的實(shí)現(xiàn)類都是誰用誰寫。*/IProducer proxyProducer = (IProducer) Proxy.newProxyInstance(producer.getClass().getClassLoader(),producer.getClass().getInterfaces(),new InvocationHandler() {/*** 作用:執(zhí)行被代理對象的任何接口方法都會經(jīng)過該方法* 方法參數(shù)的含義* @param proxy 代理對象的引用* @param method 當(dāng)前執(zhí)行的方法* @param args 當(dāng)前執(zhí)行方法所需的參數(shù)* @return 和被代理對象方法有相同的返回值* @throws Throwable*/public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//提供增強(qiáng)的代碼Object returnValue = null;//1.獲取方法執(zhí)行的參數(shù)Float money = (Float)args[0];//2.判斷當(dāng)前方法是不是銷售if("saleProduct".equals(method.getName())) {returnValue = method.invoke(producer, money*0.8f);}return returnValue;}});proxyProducer.saleProduct(10000f);} } package com.learn.proxy;/*** 對生產(chǎn)廠家要求的接口*/ public interface IProducer {/*** 銷售* @param money*/public void saleProduct(float money);/*** 售后* @param money*/public void afterService(float money); } package com.learn.proxy;/*** 一個生產(chǎn)者*/ public class Producer implements IProducer{/*** 銷售* @param money*/public void saleProduct(float money){System.out.println("銷售產(chǎn)品,并拿到錢:"+money);}/*** 售后* @param money*/public void afterService(float money){System.out.println("提供售后服務(wù),并拿到錢:"+money);} }?
總結(jié)
- 上一篇: 编写业务层和持久层事务控制代码并配置sp
- 下一篇: AOP的概念