java Proxy(代理機制)

  我們知道Spring主要有兩大思想,一個是IoC,另一個就是AOP,對于IoC,依賴注入就不用多說了,而對于Spring的核心AOP來說,我們不但要知道怎么通過AOP來滿足的我們的功能,我們更需要學習的是其底層是怎么樣的一個原理,而AOP的原理就是java的動態代理機制,所以本篇隨筆就是對java的動態機制進行一個回顧。

  在java的動態代理機制中,有兩個重要的類或接口,一個是 InvocationHandler(Interface)、另一個則是 Proxy(Class),這一個類和接口是實現我們動態代理所必須用到的。首先我們先來看看java的API幫助文檔是怎么樣對這兩個類進行描述的:

InvocationHandler:

InvocationHandler is the interface implemented by the invocation handler of a proxy instance. Each proxy instance has an associated invocation handler. When a method is invoked on a proxy instance, the method invocation is encoded and dispatched to the invoke method of its invocation handler.

每一個動態代理類都必須要實現InvocationHandler這個接口,并且每個代理類的實例都關聯到了一個handler,當我們通過代理對象調用一個方法的時候,這個方法的調用就會被轉發為由InvocationHandler這個接口的 invoke 方法來進行調用。我們來看看InvocationHandler這個接口的唯一一個方法?invoke?方法:

Object invoke(Object proxy, Method method, Object[] args) throws Throwable

我們看到這個方法一共接受三個參數,那么這三個參數分別代表什么呢?

Object invoke(Object proxy, Method method, Object[] args) throws Throwableproxy:  指代我們所代理的那個真實對象
method:  指代的是我們所要調用真實對象的某個方法的Method對象
args:  指代的是調用真實對象某個方法時接受的參數

如果不是很明白,等下通過一個實例會對這幾個參數進行更深的講解。

接下來我們來看看Proxy這個類:

Proxy provides static methods for creating dynamic proxy classes and instances, and it is also the superclass of all dynamic proxy classes created by those methods. 

Proxy這個類的作用就是用來動態創建一個代理對象的類,它提供了許多的方法,但是我們用的最多的就是?newProxyInstance?這個方法:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,  InvocationHandler h)  throws IllegalArgumentException
Returns an instance of a proxy class for the specified interfaces that dispatches method invocations to the specified invocation handler.

這個方法的作用就是得到一個動態的代理對象,其接收三個參數,我們來看看這三個參數所代表的含義:

復制代碼
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentExceptionloader:  一個ClassLoader對象,定義了由哪個ClassLoader對象來對生成的代理對象進行加載interfaces:  一個Interface對象的數組,表示的是我將要給我需要代理的對象提供一組什么接口,如果我提供了一組接口給它,那么這個代理對象就宣稱實現了該接口(多態),這樣我就能調用這組接口中的方法了h:  一個InvocationHandler對象,表示的是當我這個動態代理對象在調用方法的時候,會關聯到哪一個InvocationHandler對象上
復制代碼

好了,下面給出代理樣例:

首先我們定義了一個HjzggMethod類型的接口

package com.hjzgg.proxy;public interface HjzggMethod {public int addMethod(int a, int b);public int subMethod(int a, int b);
}

接著,定義了一個類來實現這個接口,這個類就是我們的真實對象,HjzggMethodImpl?類

package com.hjzgg.proxy;public class HjzggMethodImpl implements HjzggMethod {@Overridepublic int addMethod(int a, int b) {System.out.println("我執行了!");return a+b;}@Overridepublic int subMethod(int a, int b) {return a-b;}}

創建動態代理類,這個類并沒有實現InvocationHandler ,而是在類方法中間接的創建一個InvocationHandler 實例

package com.hjzgg.proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
import java.util.Arrays;public class MethodProxy {private HjzggMethod target;public MethodProxy(HjzggMethod target){super();this.target = target;}public HjzggMethod getMethodProxy(){HjzggMethod proxy = null;//代理對象由哪一個類加載器加載ClassLoader loader = target.getClass().getClassLoader();//代理對象的類型,即其中有哪些方法//Class[] interfaces = new Class[]{HjzggMethod.class};Class[] interfaces = target.getClass().getInterfaces();//當調用代理對象其中的方法時,該執行的代碼InvocationHandler h = new InvocationHandler(){@Overridepublic Object invoke(Object proxy, java.lang.reflect.Method method, Object[] args) throws Throwable {String methodName = method.getName();System.out.println(method);System.out.println("the method: " + methodName + "開始, 參數: "+Arrays.asList(args));Object result = method.invoke(target, args);System.out.println("the method: "+methodName+"結束, 結果: " + result);return result;}};proxy=(HjzggMethod) Proxy.newProxyInstance(loader, interfaces, h);return proxy;}
}

最后測試類如下:

package com.hjzgg.proxy;import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;public class ProxyTest {public static String getModifier(int modifier){String result = "";switch(modifier){case Modifier.PRIVATE:result = "private";case Modifier.PUBLIC:result = "public";case Modifier.PROTECTED:result = "protected";case Modifier.ABSTRACT :result = "abstract";case Modifier.FINAL :result = "final";case Modifier.NATIVE :result = "native";case Modifier.STATIC :result = "static";case Modifier.SYNCHRONIZED :result = "synchronized";case Modifier.STRICT  :result = "strict";case Modifier.TRANSIENT :result = "transient";case Modifier.VOLATILE :result = "volatile";case Modifier.INTERFACE :result = "interface";}return result;}public static void printClassDefinition(Class clz){String clzModifier = getModifier(clz.getModifiers());if(clzModifier!=null && !clzModifier.equals("")){clzModifier = clzModifier + " ";}String superClz = clz.getSuperclass().getName();if(superClz!=null && !superClz.equals("")){superClz = "extends " + superClz;}Class[] interfaces = clz.getInterfaces();String inters = "";for(int i=0; i<interfaces.length; i++){if(i==0){inters += "implements ";}inters += interfaces[i].getName();}System.out.println(clzModifier +clz.getName()+" " + superClz +" " + inters );System.out.println("{");Field[] fields = clz.getDeclaredFields();for(int i=0; i<fields.length; i++){String modifier = getModifier(fields[i].getModifiers());if(modifier!=null && !modifier.equals("")){modifier = modifier + " ";}String fieldName = fields[i].getName();String fieldType = fields[i].getType().getName();System.out.println("    "+modifier + fieldType + " "+ fieldName + ";");}System.out.println();Method[] methods = clz.getDeclaredMethods();for(int i=0; i<methods.length; i++){Method method = methods[i];String modifier = getModifier(method.getModifiers());if(modifier!=null && !modifier.equals("")){modifier = modifier + " ";}String methodName = method.getName();Class returnClz = method.getReturnType();String retrunType = returnClz.getName();Class[] clzs = method.getParameterTypes();String paraList = "(";for(int j=0; j<clzs.length; j++){paraList += clzs[j].getName();if(j != clzs.length -1 ){paraList += ", ";}}paraList += ")";clzs = method.getExceptionTypes();String exceptions = "";for(int j=0; j<clzs.length; j++){if(j==0){exceptions += "throws ";}exceptions += clzs[j].getName();if(j != clzs.length -1 ){exceptions += ", ";}}exceptions += ";";String methodPrototype = modifier +retrunType+" "+methodName+paraList+exceptions;System.out.println("    "+methodPrototype );}System.out.println("}");}public static void main(String[] args) throws SecurityException, NoSuchMethodException, IllegalArgumentException, IllegalAccessException, InvocationTargetException{HjzggMethod method = new HjzggMethodImpl();System.out.println(method.getClass().getMethod("addMethod", int.class, int.class));HjzggMethod methodProxy = new MethodProxy(method).getMethodProxy();methodProxy.addMethod(10, 20);printClassDefinition(methodProxy.getClass());
    }
}

最后的輸出結果:

public int com.hjzgg.proxy.HjzggMethodImpl.addMethod(int,int)
public abstract int com.hjzgg.proxy.HjzggMethod.addMethod(int,int)
the method: addMethod開始, 參數: [10, 20]
the method: addMethod結束, 結果: 30
$Proxy0 extends java.lang.reflect.Proxy implements com.hjzgg.proxy.HjzggMethod
{java.lang.reflect.Method m1;java.lang.reflect.Method m3;java.lang.reflect.Method m0;java.lang.reflect.Method m4;java.lang.reflect.Method m2;int subMethod(int, int);boolean equals(java.lang.Object);java.lang.String toString();int hashCode();int addMethod(int, int);
}

?

  從輸出結果來看,原來通過?HjzggMethod methodProxy = new MethodProxy(method).getMethodProxy();得到的HjzggMethod 實例其實是繼承自Proxy 的,并且實現了HjzggMethod (我們之前定義的接口)接口中的方法。在實現的方法(比如addMethod)中是通過?InvocationHandler 調用invoke方法,然后InvocationHandler的invoke方法中又調用method中的invoke來實現。執行的順序是??methodProxy.addMethod(10, 20); ->?InvocationHandler中的invoke() -> method中的invoke()。還有就是method這個對象是可以通過接口.class來獲得的。method.invoke(obj, args)中的obj實體一定是實現了該method對應的接口。恰巧我們在創建代理對象的時候,(HjzggMethod) Proxy.newProxyInstance(loader, interfaces, h),也有interfaces(接口的字節碼文件對象)。

通過下面的例子,你就可以輕松的理解InvocationHandler中invoke()方法中的method是如何得來的!

Method method = HjzggMethod.class.getMethod("addMethod", int.class, int.class);//通過HjzggMethod接口得到Methed實例

HjzggMethod hjzggMethod = new HjzggMethodImpl();//創建實現HjzggMethod接口的HjzggMethodImpl對象
method.invoke(hjzggMethod, 10, 20);//執行方法, 相當于hjzggMethod.addMethod(10, 20);

?

轉載于:https://www.cnblogs.com/hujunzheng/p/4872856.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/531406.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/531406.shtml
英文地址,請注明出處:http://en.pswp.cn/news/531406.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

(十三)linux中斷底半部分處理機制

這篇文章介紹一下linux中斷的底半部分的tasklet和workquene兩種處理機制&#xff0c;其中tasklet中不能有延時函數&#xff0c;workquene的處理函數可以加入延時操作 目錄&#xff08;一&#xff09;tasklet小任務處理機制&#xff08;1&#xff09;tasklet相關函數接口&#x…

Codeforces Round #326 (Div. 2) B. Pasha and Phone C. Duff and Weight Lifting

B. Pasha and PhonePasha has recently bought a new phone jPager and started adding his friends phone numbers there. Each phone number consists of exactly n digits. Also Pasha has a number k and two sequences of length n?/?k (n is divisible by k) a1,?a2,?…

vmware中裝的ubuntu上不了網

本文章針對橋接方式進行講解&#xff0c;如果需要另外兩種連接方式請參考文末給出的鏈接 &#xff08;一&#xff09;問題 主機和虛擬機可以相互ping通&#xff0c;但是卻不能ping網址 &#xff08;二&#xff09;解決辦法 vmware為我們提供了三種網絡工作模式&#xff0c;…

document.getElementById()與 $()區別

document.getElementById()返回的是DOM對象&#xff0c;而$()返回的是jQuery對象 什么是jQuery對象&#xff1f; ---就是通過jQuery包裝DOM對象后產生的對象。jQuery對象是jQuery獨有的&#xff0c;其可以使用jQuery里的方法。 比如&#xff1a; $("#test").html() 意…

關于gedit的編碼問題

今天由于gedit的編碼格式導致LCD顯示屏的問題&#xff0c;開始沒有想到后來才發現&#xff0c;在這記錄一下 #include <stdio.h> #include <unistd.h> #include <stdio.h> #include <fcntl.h> #include <linux/fb.h> #include <sys/mman.h>…

c語言表白程序代碼

雙十一要到了&#xff0c;好激動啊&#xff01;&#xff01;&#xff01; 是時候準備出手了&#xff01; 花了一天的時間寫的表白代碼。 表示自己弱弱的..... 看了網上好多都是js寫的&#xff0c;感覺碉堡了&#xff01;js用的不熟&#xff0c;前端不好&#xff0c;java&#x…

tiny4412移植tslib庫

1、將tslib-1.4.tar.gz拷貝到虛擬機某個路徑進行解壓 2、進入解壓路徑tslib 3、執行#./autogen.sh 如果提示&#xff1a;./autogen.sh: 4: ./autogen.sh: autoreconf: not found 原因&#xff1a;沒有安裝automake工具, 解決辦法:需要安裝此工具&#xff1a; apt-get instal…

移植QT到tiny4412開發板

目錄&#xff08;一&#xff09; 環境準備&#xff08;二&#xff09; Qt源代碼下載&#xff08;三&#xff09; 移植tslib庫&#xff08;四&#xff09;操作流程1.解壓qt源碼包2.配置編譯環境3.生成Makefile4.編譯安裝5.安裝一些庫用來支持 qt6. 添加以下內容到開發板目錄下的…

c++面試常用知識(sizeof計算類的大小,虛擬繼承,重載,隱藏,覆蓋)

一. sizeof計算結構體 注&#xff1a;本機機器字長為64位 1.最普通的類和普通的繼承 #include<iostream> using namespace std;class Parent{ public:void fun(){cout<<"Parent fun"<<endl;} }; class Child : public Parent{ public:void fun(){…

嵌入式面試題(一)

目錄1 關鍵字volatile有什么含義&#xff1f;并給出三個不同的例子2. c和c中的struct有什么不同&#xff1f;3.進程和線程區別4.ARM流水線5.使用斷言6 .嵌入式系統的定義7 局部變量能否和全局變量重名&#xff1f;8 如何引用一個已經定義過的全局變量&#xff1f;9、全局變量可…

能ping通ip但無法ping通域名和localhost //ping: bad address 'www.baidu.com'

錯誤描述&#xff1a; ~ # ping localhost ping: bad address localhost原因&#xff0c;在/etc目錄下缺少hosts文件&#xff0c;將linux中的/etc hosts文件拷入即可 ~ # ping localhost PING localhost (127.0.0.1): 56 data bytes 64 bytes from 127.0.0.1: seq0 ttl64 tim…

eclipse導入web項目之后項目中出現小紅叉解決辦法

項目中有小紅叉我遇到的最常見的情況&#xff1a; 1、項目代碼本身有問題。&#xff08;這個就不說了&#xff0c;解決錯誤就OK&#xff09; 2、項目中的jar包丟失。&#xff08;有時候eclipse打開時會出現jar包丟失的情況&#xff0c;關閉eclipse重新打開或者重新引入jar包就O…

arm開發板通過網線連接筆記本電腦上外網

需要工具&#xff1a;arm開發板&#xff0c;網線&#xff0c;一臺雙網卡的win7筆記本電腦&#xff08;筆記本電腦一般都是雙網卡&#xff09; 一、筆記本電腦需要先連上外網&#xff0c;可以連上家里的WIFI&#xff0c;或者手機開熱點&#xff08;本人未測試過連接手機的熱點&…

windows下實現Git在局域網使用

1.首先在主機A上創建一個文件夾用于存放你要公開的版本庫。然后進入這個文件夾&#xff0c;右鍵->Git create repository here&#xff0c;彈出的窗口中勾選Make it Bare&#xff01;之后將這個文件夾完全共享&#xff08;共享都會吧&#xff1f;注意權限要讓使用這個文件夾…

解決linux下QtCreator無法輸入中文的情況

安裝了QtCreator(Qt5.3.1自帶版本)后無法輸入中文&#xff0c;確切的說是無法打開輸入法。以前使用iBus輸入法的時候沒有這個問題&#xff0c;現在使用sougou輸入法才有的這個問題。 可以查看此文 http://www.cnblogs.com/oloroso/p/5114041.html 原因 有問題就得找原因&…

lintcode 滑動窗口的最大值(雙端隊列)

題目鏈接&#xff1a;http://www.lintcode.com/zh-cn/problem/sliding-window-maximum/# 滑動窗口的最大值 給出一個可能包含重復的整數數組&#xff0c;和一個大小為 k 的滑動窗口, 從左到右在數組中滑動這個窗口&#xff0c;找到數組中每個窗口內的最大值。 樣例 給出數組 [1…

你的main函數規范嗎?

在學習c語言的時候&#xff0c;有一個函數一直被我們使用&#xff0c;那就是main函數&#xff0c;但是你知道標準里面是怎么規定它的寫法嗎&#xff1f; 平時看見的main函數有下面這幾種&#xff1a; 1.int main(void){ }2.int main(){ }3.int main(int argc, char *argv[])…

lintcode 最長上升連續子序列 II(二維最長上升連續序列)

題目鏈接&#xff1a;http://www.lintcode.com/zh-cn/problem/longest-increasing-continuous-subsequence-ii/ 最長上升連續子序列 II 給定一個整數矩陣&#xff08;其中&#xff0c;有 n 行&#xff0c; m 列&#xff09;&#xff0c;請找出矩陣中的最長上升連續子序列。&a…

適用于Linux的Windows子系統WSL

以前使用的都是在虛擬機里安裝linux&#xff0c;最近才發現在win10提供了WSL(Windows Subsystem for Linux) &#xff0c;簡單來說就是可以在win10里面直接使用Linux。 &#xff08;一&#xff09;首先打開Microsoft Store , 搜索 Linux &#xff08;二&#xff09;選擇自己需…

jsp通過易寶方式實現在線支付

項目下載地址: https://github.com/hjzgg/OnlinePayment 參考&#xff1a;http://blog.csdn.net/jadyer/article/details/7380259?utm_sourcetuicool&utm_mediumreferral 效果圖1&#xff1a;請求界面 效果圖2&#xff1a;地支付請求和易寶之間建立連接之后跳轉到相應的銀…