Java基礎(chǔ)之代理原理與用法詳解
本文實例講述了Java基礎(chǔ)之代理原理與用法。分享給大家供大家參考,具體如下:
1.什么是代理
動態(tài)代理技術(shù)是整個java技術(shù)中最重要的一個技術(shù),它是學(xué)習(xí)java框架的基礎(chǔ),不會動態(tài)代理技術(shù),那么在學(xué)習(xí)Spring這些框架時是學(xué)不明白的。
動態(tài)代理技術(shù)就是用來產(chǎn)生一個對象的代理對象的。在開發(fā)中為什么需要為一個對象產(chǎn)生代理對象呢?
舉一個現(xiàn)實生活中的例子:歌星或者明星都有一個自己的經(jīng)紀(jì)人,這個經(jīng)紀(jì)人就是他們的代理人,當(dāng)我們需要找明星表演時,不能直接找到該明星,只能是找明星的代理人。比如劉德華在現(xiàn)實生活中非常有名,會唱歌,會跳舞,會拍戲,劉德華在沒有出名之前,我們可以直接找他唱歌,跳舞,拍戲,劉德華出名之后,他干的第一件事就是找一個經(jīng)紀(jì)人,這個經(jīng)紀(jì)人就是劉德華的代理人(代理),當(dāng)我們需要找劉德華表演時,不能直接找到劉德華了(劉德華說,你找我代理人商談具體事宜吧!),只能是找劉德華的代理人,因此劉德華這個代理人存在的價值就是攔截我們對劉德華的直接訪問!
這個現(xiàn)實中的例子和我們在開發(fā)中是一樣的,我們在開發(fā)中之所以要產(chǎn)生一個對象的代理對象,主要用于攔截對真實業(yè)務(wù)對象的訪問。那么代理對象應(yīng)該具有什么方法呢?代理對象應(yīng)該具有和目標(biāo)對象相同的方法
所以在這里明確代理對象的兩個概念:
1、代理對象存在的價值主要用于攔截對真實業(yè)務(wù)對象的訪問。
2、代理對象應(yīng)該具有和目標(biāo)對象(真實業(yè)務(wù)對象)相同的方法。劉德華(真實業(yè)務(wù)對象)會唱歌,會跳舞,會拍戲,我們現(xiàn)在不能直接找他唱歌,跳舞,拍戲了,只能找他的代理人(代理對象)唱歌,跳舞,拍戲,一個人要想成為劉德華的代理人,那么他必須具有和劉德華一樣的行為(會唱歌,會跳舞,會拍戲),劉德華有什么方法,他(代理人)就要有什么方法,我們找劉德華的代理人唱歌,跳舞,拍戲,但是代理人不是真的懂得唱歌,跳舞,拍戲的,真正懂得唱歌,跳舞,拍戲的是劉德華,在現(xiàn)實中的例子就是我們要找劉德華唱歌,跳舞,拍戲,那么只能先找他的經(jīng)紀(jì)人,交錢給他的經(jīng)紀(jì)人,然后經(jīng)紀(jì)人再讓劉德華去唱歌,跳舞,拍戲。
2.代理的分類
靜態(tài)代理:代理類和委托類在代碼運行之前關(guān)系就已經(jīng)確定了,也就是說在代理類的代碼一開始就已經(jīng)存在了。
動態(tài)代理:動態(tài)代理類的字節(jié)碼在程序運行的時候生成。
3.Java中的代理
"java.lang.reflect.Proxy"類介紹
現(xiàn)在要生成某一個對象的代理對象,這個代理對象通常也要編寫一個類來生成,所以首先要編寫用于生成代理對象的類。在java中如何用程序去生成一個對象的代理對象呢,java在JDK1.5之后提供了一個"java.lang.reflect.Proxy"類,通過"Proxy"類提供的一個newProxyInstance方法用來創(chuàng)建一個對象的代理對象,如下所示:
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
newProxyInstance方法用來返回一個代理對象,這個方法總共有3個參數(shù),ClassLoader loader用來指明生成代理對象使用哪個類裝載器,Class<?>[] interfaces用來指明生成哪個對象的代理對象,通過接口指定,InvocationHandler h用來指明產(chǎn)生的這個代理對象要做什么事情。所以我們只需要調(diào)用newProxyInstance方法就可以得到某一個對象的代理對象了。
編寫生成代理對象的類
在java中規(guī)定,要想產(chǎn)生一個對象的代理對象,那么這個對象必須要有一個接口,所以我們第一步就是設(shè)計這個對象的接口,在接口中定義這個對象所具有的行為(方法)
StduentInterface.java--定義對象的行為接口
public interface StduentInterface { public int insertStudent(Student student); }
StudentDao.java--定義目標(biāo)業(yè)務(wù)對象類
public class StudentDAO implements StduentInterface { @Override public int insertStudent(Student student) { // 以下這段代碼是業(yè)務(wù)邏輯 return null == student ? 0 : 1; } }
創(chuàng)建靜態(tài)代理
public class StudentStaticProxyDAO { private StudentDAO studentDAO; public StudentStaticProxyDAO(StudentDAO studentDAO) { this.studentDAO = studentDAO; } public int insertStudent(Student student) { Logger.info("開始插入一個學(xué)生記錄: " + student.toString()); int number = studentDAO.insertStudent(student); Logger.info("插入 " + number + " 條學(xué)生記錄"); return number; } }
動態(tài)代理
public class LoggerDynamicProxy implements InvocationHandler{ private Object target; public LoggerDynamicProxy(Object obj) { this.target = obj; } /** * 動態(tài)代理類中的 invoke 方法 * * 三個參數(shù)的意義: * * proxy :代理對象 * method : 委托類中要調(diào)用的方法 * args :method 方法的參數(shù) */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Logger.info("before 業(yè)務(wù)方法調(diào)用之前"); Object obj = method.invoke(target, args); Logger.info("before 業(yè)務(wù)方法調(diào)用之前"); return obj; } }
測試類
public class ProxyTest { public static void main(String[] args) { /** * 基礎(chǔ)組件準(zhǔn)備 */ StudentDAO studentDao = new StudentDAO(); Student student = new Student(100000); StudentStaticProxyDAO studentProxyDAO = new StudentStaticProxyDAO(studentDao); /** * 測試委托類的功能 */ System.out.println(studentDao.insertStudent(student)); /** * 測試靜態(tài)代理類的功能 */ System.out.println(studentProxyDAO.insertStudent(student)); /** * 測試動態(tài)代理類的功能 */ StduentInterface newProxyInstance = (StduentInterface)Proxy.newProxyInstance(studentDao.getClass().getClassLoader(), studentDao.getClass().getInterfaces(), new LoggerDynamicProxy(studentDao)); int insertStudent = newProxyInstance.insertStudent(student); System.out.println(insertStudent); }
運行結(jié)果
1
INFO: 開始插入一個學(xué)生記錄: Student [id=100000, name=null]
INFO: 插入 1 條學(xué)生記錄
1
INFO: before 業(yè)務(wù)方法調(diào)用之前
INFO: before 業(yè)務(wù)方法調(diào)用之前
1
INFO: before 業(yè)務(wù)方法調(diào)用之前
INFO: before 業(yè)務(wù)方法調(diào)用之前
1
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java面向?qū)ο蟪绦蛟O(shè)計入門與進階教程》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對大家java程序設(shè)計有所幫助。
相關(guān)文章
Java throw Exception實現(xiàn)異常轉(zhuǎn)換
這篇文章主要介紹了Java throw Exception實現(xiàn)異常轉(zhuǎn)換,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-04-04Spring mvc是如何實現(xiàn)與數(shù)據(jù)庫的前后端的連接操作的?
今天給大家?guī)淼氖顷P(guān)于Spring mvc的相關(guān)知識,文章圍繞著Spring mvc是如何實現(xiàn)與數(shù)據(jù)庫的前后端的連接操作的展開,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下2021-06-06Java定時調(diào)用.ktr文件的示例代碼(解決方案)
這篇文章主要介紹了Java定時調(diào)用.ktr文件的示例代碼,本文給大家分享遇到問題及解決方法,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-04-04Java實現(xiàn)使用Websocket發(fā)送消息詳細代碼舉例
這篇文章主要給大家介紹了關(guān)于Java實現(xiàn)使用Websocket發(fā)送消息的相關(guān)資料,WebSocket是一種協(xié)議,用于在Web應(yīng)用程序和服務(wù)器之間建立實時、雙向的通信連接,文中通過代碼介紹的非常詳細,需要的朋友可以參考下2024-05-05透徹理解Java中Synchronized(對象鎖)和Static Synchronized(類鎖)的區(qū)別
這篇文章主要介紹了Java中Synchronized(對象鎖)和Static Synchronized(類鎖)的區(qū)別,希望對大家有所幫助,一起跟隨小編過來看看吧2018-05-05