中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中的代理模式有哪些

發布時間:2021-01-14 15:40:36 來源:億速云 閱讀:155 作者:Leah 欄目:編程語言

本篇文章為大家展示了Java中的代理模式有哪些,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。

代理模式

代理(Proxy)是一種設計模式,提供了對目標對象另外的訪問方式;即通過代理對象訪問目標對象.這樣做的好處是:可以在目標對象實現的基礎上,增強額外的功能操作,即擴展目標對象的功能。

這里使用到編程中的一個思想:不要隨意去修改別人已經寫好的代碼或者方法,如果需改修改,可以通過代理的方式來擴展該方法。

舉個例子來說明代理的作用:假設我們想邀請一位明星,那么并不是直接連接明星,而是聯系明星的經紀人,來達到同樣的目的.明星就是一個目標對象,他只要負責活動中的節目,而其他瑣碎的事情就交給他的代理人(經紀人)來解決.這就是代理思想在現實中的一個例子。

代理模式的 關鍵點是:代理對象與目標對象.代理對象是對目標對象的擴展,并會調用目標對象 .

靜態代理

靜態代理在使用時,需要定義接口或者父類,被代理對象與代理對象一起實現相同的接口或者是繼承相同父類。

實例說明:

模擬保存動作,定義一個保存動作的接口:IUserDao.java,然后目標對象實現這個接口的方法UserDao.java,此時如果使用靜態代理方式,就需要在代理對象(UserDaoProxy.java)中也實現IUserDao接口.調用的時候通過調用代理對象的方法來調用目標對象。

需要 注意 的是,代理對象與目標對象要實現相同的接口,然后通過調用相同的方法來調用目標對象的方法。

接口:IUserDao.java

package net.ydstudio.service;
/**
 * @author Nick
 * @projectName javaLean
 * @package net.ydstudio.service
 * @createDate 2018/08/16 15:35
 * @updateDate 2018/08/16 15:35
 */
public interface IUserDao {
  /**
   * 保存數據庫
   * @param: []
   * @return: void
   */
  void save();
}

目標對象:UserDao.java

package net.ydstudio.service.impl;
import net.ydstudio.service.IUserDao;
/**
 * @author Nick
 * @projectName javaLean
 * @package net.ydstudio.service.impl
 * @createDate 2018/08/16 15:36
 * @updateDate 2018/08/16 15:36
 */
public class UserDao implements IUserDao {
  /**
   * 保存數據庫
   *
   * @param: []
   * @return: void
   */
  public void save() {
    System.out.println("數據已經保存到數據庫");
  }
}

代理對象:UserDaoProxy.java

package net.ydstudio.staticproxy;
import net.ydstudio.service.IUserDao;
import net.ydstudio.service.impl.UserDao;
/**
 * @author Nick
 * @projectName javaLean
 * @package net.ydstudio.staticproxy
 * @createDate 2018/08/16 15:37
 * @updateDate 2018/08/16 15:37
 */
public class UserDaoProxy implements IUserDao {
  /**
   * 保存被代理的對象
   */
  private UserDao target;
  public UserDaoProxy(UserDao target) {
    this.target = target;
  }
  /**
   * 保存數據庫
   *
   * @param: []
   * @return: void
   */
  public void save() {
    System.out.println("開始保存數據……");
    target.save();
    System.out.println("結束保存數據……");
  }
}

測試類:

package net.ydstudio.proxy;
import net.ydstudio.service.IUserDao;
import net.ydstudio.service.impl.UserDao;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import static org.junit.Assert.*;
/**
 * @author Nick
 * @projectName javaLean
 * @package net.ydstudio.proxy
 * @createDate 2018/08/16 15:58
 * @updateDate 2018/08/16 15:58
 */
@RunWith(JUnit4.class)
public class ProxyFactoryTest {
  @Test
  public void test(){
    // 目標對象
    IUserDao target = new UserDao();
    System.out.println(target.getClass());
    // 給目標對象,創建代理對象
    IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance();
    // class $Proxy()內存中動態生成的代理對象
    System.out.println(proxy.getClass());
    // 執行方法 代理對象
    proxy.save();
  }
}

靜態代理總結:

  • 可以在不修改代理目標對象的前提下,對代理目標的功能進行拓展。

  • 需要實現代理目標對象實現的接口,一旦代理目標所實現的接口有修改,目標對象與代理都需要維護。

要解決上面靜態代理的缺點,就必須使用動態代理的方式。

動態代理

動態代理有以下特點:

  • 代理對象,不需要實現接口

  • 代理對象的生成,是利用JDK的API,動態的在內存中構建代理對象(需要我們指定創建代理對象/目標對象實現的接口的類型)

  • 動態代理也叫做:JDK代理,接口代理

JDK中生成代理對象的api

JDK實現代理只需要使用靜態的newProxyInstance方法,該方法需要接收三個參數:

 public static Object newProxyInstance(ClassLoader loader,
                     Class<?>[] interfaces,
                     InvocationHandler h)
    throws IllegalArgumentException

參數按順序解釋如下:

  • ClassLoader loader,:指定當前目標對象使用類加載器,獲取加載器的方法是固定的

  • Class[] interfaces,:目標對象實現的接口的類型,使用泛型方式確認類型

  • InvocationHandler h:事件處理,執行目標對象的方法時,會觸發事件處理器的方法,會把當前執行目標對象的方法作為參數傳入

代碼實現:

接口類IUserDao.java以及接口實現類,目標對象UserDao是一樣的,沒有做修改.在這個基礎上,增加一個代理工廠類(ProxyFactory.java),將代理類寫在這個地方,然后在測試類(需要使用到代理的代碼)中先建立目標對象和代理對象的聯系,然后代用代理對象的中同名方法。

代理工廠類ProxyFactory:

package net.ydstudio.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * @author Nick
 * @projectName javaLean
 * @package net.ydstudio.proxy
 * @createDate 2018/08/16 15:44
 * @updateDate 2018/08/16 15:44
 */
public class ProxyFactory {
  /**
   * 維護一個代理的目標對象
   */
  private Object target;
  public ProxyFactory(Object target){
    this.target = target;
  }
  /**
   * 給目標對象生成代理對象
   * @param: []
   * @return: java.lang.Object
   */
  public Object getProxyInstance(){
    return Proxy.newProxyInstance(
        target.getClass().getClassLoader(),
        target.getClass().getInterfaces(),
        new InvocationHandler() {
          public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("開始事務2");
            //執行目標對象方法
            Object returnValue = method.invoke(target, args);
            System.out.println("提交事務2");
            return returnValue;
          }
        }
    );
  }
}

測試類:

package net.ydstudio.proxy;
import net.ydstudio.service.IUserDao;
import net.ydstudio.service.impl.UserDao;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import static org.junit.Assert.*;
/**
 * @author Nick
 * @projectName javaLean
 * @package net.ydstudio.proxy
 * @createDate 2018/08/16 15:58
 * @updateDate 2018/08/16 15:58
 */
@RunWith(JUnit4.class)
public class ProxyFactoryTest {
  @Test
  public void test(){
    // 目標對象
    IUserDao target = new UserDao();
    System.out.println(target.getClass());
    // 給目標對象,創建代理對象
    IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance();
    // class $Proxy()內存中動態生成的代理對象
    System.out.println(proxy.getClass());

    // 執行方法 代理對象
    proxy.save();
  }
}

JDK實現代理總結:代理對象不需要實現接口,但是目標對象一定要實現接口,否則不能用動態代理。

上述內容就是Java中的代理模式有哪些,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

韶关市| 南乐县| 新巴尔虎左旗| 吉安市| 河池市| 长乐市| 德兴市| 阿图什市| 山东省| 视频| 黑水县| 阳泉市| 泊头市| 宁化县| 息烽县| 民乐县| 开鲁县| 和平县| 多伦县| 松江区| 余姚市| 赣榆县| 涞水县| 托克托县| 普兰店市| 木兰县| 武安市| 东乡族自治县| 昌图县| 左权县| 达孜县| 双峰县| 上虞市| 留坝县| 句容市| 丹凤县| 花垣县| 突泉县| 新蔡县| 武定县| 手机|