主頁 > 後端開發 > Java設計模式-代理模式

Java設計模式-代理模式

2021-02-12 10:52:20 後端開發

目錄

  • 1.概述
  • 2.結構
  • 3 靜態代理
  • 4 JDK動態代理
  • 5 CGLIB動態代理
  • 6 三種代理的對比
  • 7 優缺點
  • 8 使用場景

1.概述

由于某些原因需要給某物件提供一個代理以控制對該物件的訪問,這時,訪問物件不適合或者不能直接參考目標物件,代理物件作為訪問物件和目標物件之間的中介,
在這里插入圖片描述

Java中的代理按照代理類生成時機不同又分為靜態代理和動態代理靜態代理代理類在編譯期就生成,而動態代理代理類則是在Java運行時動態生成,動態代理又有JDK代理和CGLib代理兩種

2.結構

代理(Proxy)模式分為三種角色:

  • 抽象主題(Subject)類: 通過介面或抽象類宣告真實主題和代理物件實作的業務方法,(如果我們有多個真實主題類,就需要抽象成一個抽象主題)
  • 真實主題(Real Subject)類: 實作了抽象主題中的具體業務,是代理物件所代表的真實物件,是最終要參考的物件,(比如上面的聯想公司)
  • 代理(Proxy)類 : 提供了與真實主題相同的介面,其內部含有對真實主題的參考,它可以訪問、控制或擴展真實主題的功能,(地方代理商)

3 靜態代理

我們通過案例來感受一下靜態代理,

【例】火車站賣票

如果要買火車票的話,需要去火車站買票,坐車到火車站,排隊等一系列的操作,顯然比較麻煩,而火車站在多個地方都有代售點,我們去代售點買票就方便很多了,這個例子其實就是典型的代理模式,火車站是目標物件,代售點是代理物件,類圖如下:
在這里插入圖片描述
代碼如下:

//賣票介面--抽象主題類
public interface SellTickets {
    void sell();
}

//火車站  火車站具有賣票功能,所以需要實作SellTickets介面
//具體主題類
public class TrainStation implements SellTickets {

    public void sell() {
        System.out.println("火車站賣票");
    }
}

//代售點
//代理類
public class ProxyPoint implements SellTickets {
	//宣告火車站類物件
    private TrainStation station = new TrainStation();

    public void sell() {
        System.out.println("代理點收取一些服務費用");
        station.sell();			//代售點賣票實際上還是呼叫了火車找賣票的作業
    }
}

//測驗類
public class Client {
    public static void main(String[] args) {
    //創建代售點物件
        ProxyPoint pp = new ProxyPoint();
        pp.sell();				//呼叫方法進行賣票
    }
}

從上面代碼中可以看出測驗類直接訪問的是ProxyPoint類物件,也就是說ProxyPoint作為訪問物件和目標物件的中介,同時也對sell方法進行了增強(代理點收取一些服務費用),

4 JDK動態代理

接下來我們使用動態代理實作上面案例,先說說JDK提供的動態代理,Java中提供了一個動態代理類Proxy,Proxy并不是我們上述所說的代理物件的類,而是提供了一個創建代理物件的靜態方法(newProxyInstance方法)來獲取代理物件
代理類是程式在運行程序中動態的在記憶體中生成的類,

代碼如下:

//賣票介面
public interface SellTickets {
    void sell();
}

//火車站  火車站具有賣票功能,所以需要實作SellTickets介面
public class TrainStation implements SellTickets {

    public void sell() {
        System.out.println("火車站賣票");
    }
}

//代理工廠,用來創建代理物件,是獲取代理物件的工廠類
//代理類也實作了對應的介面
public class ProxyFactory {
	//宣告目標物件
    private TrainStation station = new TrainStation();
	//獲取代理物件的方法
    public SellTickets getProxyObject() {
        //使用Proxy獲取代理物件
        /*
            newProxyInstance()方法引數說明:
                ClassLoader loader : 類加載器,用于加載代理類,使用真實物件的類加載器即可
                Class<?>[] interfaces : 真實物件所實作的介面,代理模式真實物件和代理物件實作相同的介面,
                而這個是代理類實作的介面的位元組碼物件
                InvocationHandler h : 代理物件的呼叫處理程式
         */
        SellTickets sellTickets = (SellTickets) Proxy.newProxyInstance(station.getClass().getClassLoader(),
                station.getClass().getInterfaces(),
                new InvocationHandler() {
                    /*
                        InvocationHandler中invoke方法引數說明:
                            proxy : 代理物件,和sellTickets物件是同一個人物件,在invoke方法中基本不用
                            method : 對應于在代理物件上呼叫的介面方法的 Method 實體
                            args : 代理物件呼叫介面方法時傳遞的實際引數
                            回傳值:就是方法的回傳值
                     */
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

                        System.out.println("代理點收取一些服務費用(JDK動態代理方式)");
                        //執行真實物件
                        Object result = method.invoke(station, args);
                        return result;
                    }
                });
        return sellTickets;
    }
}

//測驗類
public class Client {
    public static void main(String[] args) {
        //1.創建代理工廠物件
        ProxyFactory factory = new ProxyFactory();
        //2.使用factory物件的方法獲取代理物件
        SellTickets proxyObject = factory.getProxyObject();
        //3.呼叫代理物件的方法,實際上呼叫的是invoke方法
        proxyObject.sell();
    }
}

使用了動態代理,我們思考下面問題:

  • ProxyFactory是代理類嗎?

    ProxyFactory不是代理模式中所說的代理類,而代理類是程式在運行程序中動態的在記憶體中生成的類,通過阿里巴巴開源的 Java 診斷工具(Arthas【阿爾薩斯】)查看代理類的結構:

  package com.sun.proxy;
  
  import com.itheima.proxy.dynamic.jdk.SellTickets;
  import java.lang.reflect.InvocationHandler;
  import java.lang.reflect.Method;
  import java.lang.reflect.Proxy;
  import java.lang.reflect.UndeclaredThrowableException;
  
  public final class $Proxy0 extends Proxy implements SellTickets {
      private static Method m1;
      private static Method m2;
      private static Method m3;
      private static Method m0;
  
      public $Proxy0(InvocationHandler invocationHandler) {
          super(invocationHandler);
      }
  
      static {
          try {
              m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
              m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
              m3 = Class.forName("com.itheima.proxy.dynamic.jdk.SellTickets").getMethod("sell", new Class[0]);
              m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
              return;
          }
          catch (NoSuchMethodException noSuchMethodException) {
              throw new NoSuchMethodError(noSuchMethodException.getMessage());
          }
          catch (ClassNotFoundException classNotFoundException) {
              throw new NoClassDefFoundError(classNotFoundException.getMessage());
          }
      }
  
      public final boolean equals(Object object) {
          try {
              return (Boolean)this.h.invoke(this, m1, new Object[]{object});
          }
          catch (Error | RuntimeException throwable) {
              throw throwable;
          }
          catch (Throwable throwable) {
              throw new UndeclaredThrowableException(throwable);
          }
      }
  
      public final String toString() {
          try {
              return (String)this.h.invoke(this, m2, null);
          }
          catch (Error | RuntimeException throwable) {
              throw throwable;
          }
          catch (Throwable throwable) {
              throw new UndeclaredThrowableException(throwable);
          }
      }
  
      public final int hashCode() {
          try {
              return (Integer)this.h.invoke(this, m0, null);
          }
          catch (Error | RuntimeException throwable) {
              throw throwable;
          }
          catch (Throwable throwable) {
              throw new UndeclaredThrowableException(throwable);
          }
      }
  
      public final void sell() {
          try {
              this.h.invoke(this, m3, null);
              return;
          }
          catch (Error | RuntimeException throwable) {
              throw throwable;
          }
          catch (Throwable throwable) {
              throw new UndeclaredThrowableException(throwable);
          }
      }
  }

從上面的類中,我們可以看到以下幾個資訊:

  • 代理類($Proxy0)實作了SellTickets,這也就印證了我們之前說的真實類和代理類實作同樣的介面

  • 代理類($Proxy0)將我們提供了的匿名內部類物件傳遞給了父類

  • 動態代理的執行流程是什么樣?

    下面是摘取的重點代碼:

  //程式運行程序中動態生成的代理類
  public final class $Proxy0 extends Proxy implements SellTickets {
      private static Method m3;
  		//有參構造
      public $Proxy0(InvocationHandler invocationHandler) {
          super(invocationHandler);
      }
  	//加載類com.itheima.proxy.dynamic.jdk.SellTickets,然后再去獲取方法物件
      static {
          m3 = Class.forName("com.itheima.proxy.dynamic.jdk.SellTickets").getMethod("sell", new Class[0]);
      }
  
      public final void sell() {
          this.h.invoke(this, m3, null);
      }
  }
  
  //Java提供的動態代理相關類
  public class Proxy implements java.io.Serializable {
  	protected InvocationHandler h;
  	 
  	protected Proxy(InvocationHandler h) {
          this.h = h;
      }
  }
  
  //代理工廠類
  public class ProxyFactory {
  
      private TrainStation station = new TrainStation();
  
      public SellTickets getProxyObject() {
          SellTickets sellTickets = (SellTickets) Proxy.newProxyInstance(station.getClass().getClassLoader(),
                  station.getClass().getInterfaces(),
                  new InvocationHandler() {
                      
                      public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  
                          System.out.println("代理點收取一些服務費用(JDK動態代理方式)");
                          Object result = method.invoke(station, args);
                          return result;
                      }
                  });
          return sellTickets;
      }
  }
  
  
  //測驗訪問類
  public class Client {
      public static void main(String[] args) {
          //獲取代理物件
          ProxyFactory factory = new ProxyFactory();
          SellTickets proxyObject = factory.getProxyObject();
          proxyObject.sell();
      }
  }

執行流程如下:

  1. 在測驗類中通過代理物件呼叫sell()方法
  2. 根據多型的特性,執行的是代理類($Proxy0)中的sell()方法(父類參考指向子類物件)
  3. 代理類($Proxy0)中的sell()方法中又呼叫了InvocationHandler介面的子實作類物件的invoke方法
  4. invoke方法通過反射執行了真實物件所屬類(TrainStation)中的sell()方法

5 CGLIB動態代理

同樣是上面的案例,我們再次使用CGLIB代理實作,

如果沒有定義SellTickets介面,只定義了TrainStation(火車站類),很顯然JDK代理是無法使用了,因為JDK動態代理要求必須定義介面,對介面進行代理,

CGLIB是一個功能強大,高性能的代碼生成包,它為沒有實作介面的類提供代理,為JDK的動態代理提供了很好的補充,

CGLIB是第三方提供的包,所以需要引入jar包的坐標:(pom.xnl檔案中)

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>2.2.2</version>
</dependency>

代碼如下:

//火車站
public class TrainStation {

    public void sell() {
        System.out.println("火車站賣票");
    }
}

//代理工廠,用來獲取代理物件
public class ProxyFactory implements MethodInterceptor {

    private TrainStation target = new TrainStation();

    public TrainStation getProxyObject() {
        //創建Enhancer物件,類似于JDK動態代理的Proxy類,下一步就是設定幾個引數
        Enhancer enhancer =new Enhancer();
        //設定父類的位元組碼物件
        enhancer.setSuperclass(target.getClass());
        //設定回呼函式
        enhancer.setCallback(this);
        //創建代理物件
        TrainStation obj = (TrainStation) enhancer.create();
        return obj;
    }

    /*
        intercept方法引數說明:
            o : 代理物件
            method : 真實物件中的方法的Method實體
            args : 實際引數
            methodProxy :代理物件中的方法的method實體
     */
    public TrainStation intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("代理點收取一些服務費用(CGLIB動態代理方式)");
        TrainStation result = (TrainStation) methodProxy.invokeSuper(o, args);
        return result;
    }
}

//測驗類
public class Client {
    public static void main(String[] args) {
        //創建代理工廠物件
        ProxyFactory factory = new ProxyFactory();
        //獲取代理物件
        TrainStation proxyObject = factory.getProxyObject();

        proxyObject.sell();
    }
}

6 三種代理的對比

  • jdk代理和CGLIB代理

    使用CGLib實作動態代理,CGLib底層采用ASM位元組碼生成框架,使用位元組碼技術生成代理類,在JDK1.6之前比使用Java反射效率要高,唯一需要注意的是,CGLib不能對宣告為final的類或者方法進行代理,因為CGLib原理是動態生成被代理類的子類,

    在JDK1.6、JDK1.7、JDK1.8逐步對JDK動態代理優化之后,在呼叫次數較少的情況下,JDK代理效率高于CGLib代理效率,只有當進行大量呼叫的時候,JDK1.6和JDK1.7比CGLib代理效率低一點,但是到JDK1.8的時候,JDK代理效率高于CGLib代理,所以如果有介面使用JDK動態代理,如果沒有介面使用CGLIB代理,

  • 動態代理和靜態代理

    動態代理與靜態代理相比較,最大的好處是介面中宣告的所有方法都被轉移到呼叫處理器一個集中的方法中處理(InvocationHandler.invoke),這樣,在介面方法數量比較多的時候,我們可以進行靈活處理,而不需要像靜態代理那樣每一個方法進行中轉,

    如果介面增加一個方法,靜態代理模式除了所有實作類需要實作這個方法外,所有代理類也需要實作此方法,增加了代碼維護的復雜度,而動態代理不會出現該問題

7 優缺點

優點:

  • 代理模式在客戶端與目標物件之間起到一個中介作用和保護目標物件的作用;
  • 代理物件可以擴展目標物件的功能;
  • 代理模式能將客戶端與目標物件分離,在一定程度上降低了系統的耦合度;

缺點:

  • 增加了系統的復雜度;

8 使用場景

  • 遠程(Remote)代理

    本地服務通過網路請求遠程服務,為了實作本地到遠程的通信,我們需要實作網路通信,處理其中可能的例外,為良好的代碼設計和可維護性,我們將網路通信部分隱藏起來,只暴露給本地服務一個介面,通過該介面即可訪問遠程服務提供的功能,而不必過多關心通信部分的細節,

  • 防火墻(Firewall)代理

    當你將瀏覽器配置成使用代理功能時,防火墻就將你的瀏覽器的請求轉給互聯網;當互聯網回傳回應時,代理服務器再把它轉給你的瀏覽器,

  • 保護(Protect or Access)代理

    控制對一個物件的訪問,如果需要,可以給不同的用戶提供不同級別的使用權限,

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/258938.html

標籤:java

上一篇:Java常用類筆記

下一篇:自學Java網路爬蟲-Day1

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more