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

java動態(tài)代理(jdk與cglib)詳細解析

 更新時間:2013年09月11日 08:42:40   作者:  
靜態(tài)代理:由程序員創(chuàng)建或特定工具自動生成源代碼,再對其編譯。在程序運行前,代理類的.class文件就已經(jīng)存在了

JAVA的動態(tài)代理

代理模式
代理模式是常用的java設(shè)計模式,他的特征是代理類與委托類有同樣的接口,代理類主要負責為委托類預(yù)處理消息、過濾消息、把消息轉(zhuǎn)發(fā)給委托類,以及事后處理消息等。代理類與委托類之間通常會存在關(guān)聯(lián)關(guān)系,一個代理類的對象與一個委托類的對象關(guān)聯(lián),代理類的對象本身并不真正實現(xiàn)服務(wù),而是通過調(diào)用委托類的對象的相關(guān)方法,來提供特定的服務(wù)。
按照代理的創(chuàng)建時期,代理類可以分為兩種。

靜態(tài)代理:由程序員創(chuàng)建或特定工具自動生成源代碼,再對其編譯。在程序運行前,代理類的.class文件就已經(jīng)存在了。

動態(tài)代理:在程序運行時,運用反射機制動態(tài)創(chuàng)建而成。 

首先看一下靜態(tài)代理:
1、Count.java

復制代碼 代碼如下:

package net.battier.dao; 

/**
 * 定義一個賬戶接口
 * 
 * @author Administrator
 * 
 */ 
public interface Count { 
    // 查看賬戶方法 
    public void queryCount(); 

    // 修改賬戶方法 
    public void updateCount(); 



2、CountImpl.java
復制代碼 代碼如下:

package net.battier.dao.impl; 

import net.battier.dao.Count; 

/**
 * 委托類(包含業(yè)務(wù)邏輯)
 * 
 * @author Administrator
 * 
 */ 
public class CountImpl implements Count { 

    @Override 
    public void queryCount() { 
        System.out.println("查看賬戶方法..."); 

    } 

    @Override 
    public void updateCount() { 
        System.out.println("修改賬戶方法..."); 

    } 



CountProxy.java 
package net.battier.dao.impl; 

import net.battier.dao.Count; 

/**
 * 這是一個代理類(增強CountImpl實現(xiàn)類)
 * 
 * @author Administrator
 * 
 */ 
public class CountProxy implements Count { 
    private CountImpl countImpl; 

    /**
     * 覆蓋默認構(gòu)造器
     * 
     * @param countImpl
     */ 
    public CountProxy(CountImpl countImpl) { 
        this.countImpl = countImpl; 
    } 

    @Override 
    public void queryCount() { 
        System.out.println("事務(wù)處理之前"); 
        // 調(diào)用委托類的方法; 
        countImpl.queryCount(); 
        System.out.println("事務(wù)處理之后"); 
    } 

    @Override 
    public void updateCount() { 
        System.out.println("事務(wù)處理之前"); 
        // 調(diào)用委托類的方法; 
        countImpl.updateCount(); 
        System.out.println("事務(wù)處理之后"); 

    } 



3、TestCount.java
復制代碼 代碼如下:

package net.battier.test; 

import net.battier.dao.impl.CountImpl; 
import net.battier.dao.impl.CountProxy; 

/**
 *測試Count類
 * 
 * @author Administrator
 * 
 */ 
public class TestCount { 
    public static void main(String[] args) { 
        CountImpl countImpl = new CountImpl(); 
        CountProxy countProxy = new CountProxy(countImpl); 
        countProxy.updateCount(); 
        countProxy.queryCount(); 

    } 


觀察代碼可以發(fā)現(xiàn)每一個代理類只能為一個接口服務(wù),這樣一來程序開發(fā)中必然會產(chǎn)生過多的代理,而且,所有的代理操作除了調(diào)用的方法不一樣之外,其他的操作都一樣,則此時肯定是重復代碼。解決這一問題最好的做法是可以通過一個代理類完成全部的代理功能,那么此時就必須使用動態(tài)代理完成。

再來看一下動態(tài)代理:
JDK動態(tài)代理中包含一個類和一個接口:
InvocationHandler接口:
public interface InvocationHandler {
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable;
}
參數(shù)說明:
Object proxy:指被代理的對象。
Method method:要調(diào)用的方法
Object[] args:方法調(diào)用時所需要的參數(shù)

可以將InvocationHandler接口的子類想象成一個代理的最終操作類,替換掉ProxySubject。

Proxy類:
Proxy類是專門完成代理的操作類,可以通過此類為一個或多個接口動態(tài)地生成實現(xiàn)類,此類提供了如下的操作方法:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,
InvocationHandler h)
                               throws IllegalArgumentException
參數(shù)說明:
ClassLoader loader:類加載器
Class<?>[] interfaces:得到全部的接口
InvocationHandler h:得到InvocationHandler接口的子類實例

Ps:類加載器
在Proxy類中的newProxyInstance()方法中需要一個ClassLoader類的實例,ClassLoader實際上對應(yīng)的是類加載器,在Java中主要有一下三種類加載器;
Booststrap ClassLoader:此加載器采用C++編寫,一般開發(fā)中是看不到的;
Extendsion ClassLoader:用來進行擴展類的加載,一般對應(yīng)的是jre\lib\ext目錄中的類;
AppClassLoader:(默認)加載classpath指定的類,是最常使用的是一種加載器。

動態(tài)代理
與靜態(tài)代理類對照的是動態(tài)代理類,動態(tài)代理類的字節(jié)碼在程序運行時由Java反射機制動態(tài)生成,無需程序員手工編寫它的源代碼。動態(tài)代理類不僅簡化了編程工作,而且提高了軟件系統(tǒng)的可擴展性,因為Java 反射機制可以生成任意類型的動態(tài)代理類。java.lang.reflect 包中的Proxy類和InvocationHandler 接口提供了生成動態(tài)代理類的能力。

動態(tài)代理示例:
1、BookFacade.java

復制代碼 代碼如下:

package net.battier.dao; 

public interface BookFacade { 
    public void addBook(); 
}

2、BookFacadeImpl.java
復制代碼 代碼如下:

package net.battier.dao.impl; 

import net.battier.dao.BookFacade; 

public class BookFacadeImpl implements BookFacade { 

    @Override 
    public void addBook() { 
        System.out.println("增加圖書方法。。。"); 
    } 



BookFacadeProxy.java 

package net.battier.proxy; 

import java.lang.reflect.InvocationHandler; 
import java.lang.reflect.Method; 
import java.lang.reflect.Proxy; 

/**
 * JDK動態(tài)代理代理類
 * 
 * @author student
 * 
 */ 
public class BookFacadeProxy implements InvocationHandler { 
    private Object target; 
    /**
     * 綁定委托對象并返回一個代理類
     * @param target
     * @return
     */ 
    public Object bind(Object target) { 
        this.target = target; 
        //取得代理對象 
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), 
                target.getClass().getInterfaces(), this);   //要綁定接口(這是一個缺陷,cglib彌補了這一缺陷) 
    } 

    @Override 
    /**
     * 調(diào)用方法
     */ 
    public Object invoke(Object proxy, Method method, Object[] args) 
            throws Throwable { 
        Object result=null; 
        System.out.println("事物開始"); 
        //執(zhí)行方法 
        result=method.invoke(target, args); 
        System.out.println("事物結(jié)束"); 
        return result; 
    } 



3、TestProxy.java
復制代碼 代碼如下:

package net.battier.test; 

import net.battier.dao.BookFacade; 
import net.battier.dao.impl.BookFacadeImpl; 
import net.battier.proxy.BookFacadeProxy; 

public class TestProxy { 

    public static void main(String[] args) { 
        BookFacadeProxy proxy = new BookFacadeProxy(); 
        BookFacade bookProxy = (BookFacade) proxy.bind(new BookFacadeImpl()); 
        bookProxy.addBook(); 
    } 



但是,JDK的動態(tài)代理依靠接口實現(xiàn),如果有些類并沒有實現(xiàn)接口,則不能使用JDK代理,這就要使用cglib動態(tài)代理了。

Cglib動態(tài)代理
JDK的動態(tài)代理機制只能代理實現(xiàn)了接口的類,而不能實現(xiàn)接口的類就不能實現(xiàn)JDK的動態(tài)代理,cglib是針對類來實現(xiàn)代理的,他的原理是對指定的目標類生成一個子類,并覆蓋其中方法實現(xiàn)增強,但因為采用的是繼承,所以不能對final修飾的類進行代理。

示例
1、BookFacadeCglib.java

復制代碼 代碼如下:

package net.battier.dao; 

public interface BookFacade { 
    public void addBook(); 


2、BookCadeImpl1.java
復制代碼 代碼如下:

package net.battier.dao.impl; 

/**
 * 這個是沒有實現(xiàn)接口的實現(xiàn)類
 * 
 * @author student
 * 
 */ 
public class BookFacadeImpl1 { 
    public void addBook() { 
        System.out.println("增加圖書的普通方法..."); 
    } 


3、BookFacadeProxy.java
復制代碼 代碼如下:

package net.battier.proxy; 

import java.lang.reflect.Method; 

import net.sf.cglib.proxy.Enhancer; 
import net.sf.cglib.proxy.MethodInterceptor; 
import net.sf.cglib.proxy.MethodProxy; 

/**
 * 使用cglib動態(tài)代理
 * 
 * @author student
 * 
 */ 
public class BookFacadeCglib implements MethodInterceptor { 
    private Object target; 

    /**
     * 創(chuàng)建代理對象
     * 
     * @param target
     * @return
     */ 
    public Object getInstance(Object target) { 
        this.target = target; 
        Enhancer enhancer = new Enhancer(); 
        enhancer.setSuperclass(this.target.getClass()); 
        // 回調(diào)方法 
        enhancer.setCallback(this); 
        // 創(chuàng)建代理對象 
        return enhancer.create(); 
    } 

    @Override 
    // 回調(diào)方法 
    public Object intercept(Object obj, Method method, Object[] args, 
            MethodProxy proxy) throws Throwable { 
        System.out.println("事物開始"); 
        proxy.invokeSuper(obj, args); 
        System.out.println("事物結(jié)束"); 
        return null; 

 
    } 



4、TestCglib.java
復制代碼 代碼如下:

package net.battier.test; 

import net.battier.dao.impl.BookFacadeImpl1; 
import net.battier.proxy.BookFacadeCglib; 

public class TestCglib { 

    public static void main(String[] args) { 
        BookFacadeCglib cglib=new BookFacadeCglib(); 
        BookFacadeImpl1 bookCglib=(BookFacadeImpl1)cglib.getInstance(new BookFacadeImpl1()); 
        bookCglib.addBook(); 
    } 

相關(guān)文章

  • 使用log4j2打印mybatis的sql執(zhí)行日志方式

    使用log4j2打印mybatis的sql執(zhí)行日志方式

    這篇文章主要介紹了使用log4j2打印mybatis的sql執(zhí)行日志方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • maven配置阿里倉庫的方法步驟

    maven配置阿里倉庫的方法步驟

    這篇文章主要介紹了maven配置阿里倉庫的方法步驟,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-12-12
  • Java內(nèi)部類與匿名內(nèi)部類

    Java內(nèi)部類與匿名內(nèi)部類

    這篇文章主要介紹了Java內(nèi)部類與匿名內(nèi)部類,內(nèi)部類可以直接訪問外部類的成員,包括私有成員。外部類要訪問內(nèi)部類的成員,必須要建立內(nèi)部類的對象,更多相關(guān)內(nèi)容可以參考下面文章內(nèi)容
    2022-06-06
  • Java中的同步與異步詳細介紹

    Java中的同步與異步詳細介紹

    這篇文章主要介紹了Java中的同步與異步詳細介紹,具有一定參考價值,需要的朋友可以了解下。
    2017-11-11
  • spring boot項目application.properties文件存放及使用介紹

    spring boot項目application.properties文件存放及使用介紹

    這篇文章主要介紹了spring boot項目application.properties文件存放及使用介紹,我們的application.properties文件中會有很多敏感信息,大家在使用過程中要多加小心
    2021-06-06
  • Java CAS底層實現(xiàn)原理實例詳解

    Java CAS底層實現(xiàn)原理實例詳解

    這篇文章主要介紹了Java CAS底層實現(xiàn)原理實例詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-01-01
  • SpringIOC控制反轉(zhuǎn)的原理詳解

    SpringIOC控制反轉(zhuǎn)的原理詳解

    這篇文章主要介紹了SpringIOC控制反轉(zhuǎn)的原理詳解,本來管理業(yè)務(wù)對象(bean)的操作是由我們程序員去做的,但是有了?Spring?核心容器后,這些?Bean?對象的創(chuàng)建和管理交給我們Spring容器去做了,也就是控制權(quán)由程序員變成了容器,需要的朋友可以參考下
    2023-08-08
  • JDK9為何要將String的底層實現(xiàn)由char[]改成了byte[]

    JDK9為何要將String的底層實現(xiàn)由char[]改成了byte[]

    String 類的源碼已經(jīng)由?char[]?優(yōu)化為了?byte[]?來存儲字符串內(nèi)容,為什么要這樣做呢?本文就詳細的介紹一下,感興趣的可以了解一下
    2022-03-03
  • struts1實現(xiàn)簡單的登錄功能實例(附源碼)

    struts1實現(xiàn)簡單的登錄功能實例(附源碼)

    本篇文章主要介紹了struts1實現(xiàn)簡單的登錄功能實例(附源碼),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • Spring中@RefreshScope注解的處理方法詳解

    Spring中@RefreshScope注解的處理方法詳解

    這篇文章主要介紹了Spring中@RefreshScope注解的處理方法詳解,spring啟動時會調(diào)用ClassPathBeanDefinitionScanner.java類中的doScan()對包路徑下的所有class進行掃描,獲取bean的定義,同時對bean的@RefreshScope(@Scope的父類)進行處理,需要的朋友可以參考下
    2023-10-10

最新評論