【設計模式】抽象工廠模式 Abstract Factory Pattern

来源:https://www.cnblogs.com/vaiyanzi/archive/2018/07/30/9388665.html
-Advertisement-
Play Games

簡單工廠模式是一個工廠類根據工廠方法的參數創建不出不同的產品, 工廠方法模式是每一個產品都有一個一一對應的工廠負責創建該產品。那麼今天要講的抽象工廠模式是一個工廠能夠產生關聯的一系列產品。抽象工廠模式相對於簡單工廠和工廠方法模式來著更具抽象性。 一、抽象工廠模式演繹 我們先來看一個簡單的需求: 甲方 ...


簡單工廠模式是一個工廠類根據工廠方法的參數創建不出不同的產品, 工廠方法模式是每一個產品都有一個一一對應的工廠負責創建該產品。那麼今天要講的抽象工廠模式是一個工廠能夠產生關聯的一系列產品。抽象工廠模式相對於簡單工廠和工廠方法模式來著更具抽象性。

一、抽象工廠模式演繹

我們先來看一個簡單的需求: 甲方要開發一套辦公自動化軟體,其中有一個非常重要的功能就是要能夠導入Word 文檔和Excel 文檔。

開發人員拿到需求後就開始編碼了,  很快代碼寫完了:

public  class ImportTool
{
    public void ImportWord()
    {
        Console.WriteLine("Import Word");
    }

    public void ImportExcel()
    {
        Console.WriteLine("Import Excel");
    }
}

客戶端調用代碼:

class Program
{
    static void Main(string[] args)
    {
        ImportTool importTool = new ImportTool();
        importTool.ImportWord();
        importTool.ImportExcel();

        Console.ReadKey();
    }
}

輸出結果:

image

看起來不錯, 但是看代碼,客戶端的代碼和具體的實現之間是直接new出來的 ,簡直就是面向具體編程,沒有介面沒有抽象,是不是違背了ISP原則了?,那好開發人員決定提出一個抽象層,提出一個抽象的文檔介面,   用簡單工廠模式來實現這個需求:

UML 圖如下:

 image

代碼如下:

public interface IDocument
{
    void Import();
}    
public class WordDocument : IDocument
{    
    public void Import()
    {
       Console.WriteLine("Import Word");
    }
}
public class ExcelDocument : IDocument
{
    public void Import()
    {
       Console.WriteLine("Import Excel");
    }
}

public class DocumentFactory
{
    public static IDocument Create(string documentType)
    {
        IDocument document;
        switch (documentType)
        {
            case "word":
                document = new WordDocument();
                break;
            case "excel":
                document = new ExcelDocument();
                break;
            default:
                throw new ArgumentException("Invalid argument: documentType");
        }

        return document;
    }
}

客戶端調用:

static void Main(string[] args)
{
    IDocument document=DocumentFactory.Create("word");
    document.Import();
        
    document=DocumentFactory.Create("excel");
    document.Import();

    Console.ReadKey();
}

輸出:

image

看起來完美這是一個標準的靜態工廠模式的實現。

第一次需求變更: 增加 對Power Point 的導入支持

 

 

增加了一個產品相當於,因為之前應用了簡單工廠模式現在改起來很簡單。

UML 圖:

image

代碼:

public interface IDocument
{
    void Import();
}    
public class WordDocument : IDocument
{    
    public void Import()
    {
 	        Console.WriteLine("Import Word");
    }
}
public class ExcelDocument : IDocument
{
        public void Import()
    {
 	    Console.WriteLine("Import Excel");
    }
}
public class PowerPointDocument : IDocument
{
    public void Import()
    {
        Console.WriteLine("Import Power Point");
    }
}

public class DocumentFactory
{
    public static IDocument Create(string documentType)
    {
        IDocument document;
        switch (documentType)
        {
            case "word":
                document = new WordDocument();
                break;
            case "excel":
                document = new ExcelDocument();
                break;
            case "powerpoint":
                document = new PowerPointDocument();
                break;
            default:
                throw new ArgumentException("Invalid argument: documentType");
        }

        return document;
    }
}

客戶端調用:

static void Main(string[] args)
{
    IDocument document=DocumentFactory.Create("word");
    document.Import();
        
    document=DocumentFactory.Create("excel");
    document.Import();

    document = DocumentFactory.Create("powerpoint");
    document.Import();
    Console.ReadKey();
}

輸出結果:

image

 

沒問題,一起都在控制之中。

第二次需求變更: 支持office 2007 以後的文檔格式。

Offcie 2007是個坎,2007之前的文檔格式和2007以後的文檔格式不一 樣, word 在2007前的文檔尾碼是.doc, 2007 及以後就變成.docx了這個看似簡單的需求實則是增加了小一半的工作量啊,新的格式的文檔要重新進行解碼才能拿到正確的數據,拿剛剛實現的簡單工廠設計模式來應對這次變更就要新增加3個產品類並且要工廠來創建創建者三個類的實例。看來靜態工廠已經不能再適應這一次的需求變化了,會導致靜態工廠方法的邏輯變得異常複雜難以維護。那用工廠方法模式來解決這個問題,工廠方法剛好可以將產品的創建工作提取到單獨的工廠中去完成,重構下工廠將靜態工廠模式替換成工廠方法模式:

UML 圖:

image 代碼:

public interface IDocument
{
    void Import();
}
public interface IDocumentFactory
{
    IDocument Create();
}
public class WordDocument : IDocument
{    
    public void Import()
    {
 	        Console.WriteLine("Import Word");
    }
}
public class ExcelDocument : IDocument
{
        public void Import()
    {
 	    Console.WriteLine("Import Excel");
    }
}
public class PowerPointDocument : IDocument
{
    public void Import()
    {
        Console.WriteLine("Import Power Point");
    }
}

public class WordXDocument : IDocument
{
    public void Import()
    {
        Console.WriteLine("Import WordX");
    }
}
public class ExcelXDocument : IDocument
{
    public void Import()
    {
        Console.WriteLine("Import ExcelX");
    }
}
public class PowerPointXDocument : IDocument
{
    public void Import()
    {
        Console.WriteLine("Import Power PointX");
    }
}

public class WordDocumentFactory : IDocumentFactory
{
    public  IDocument Create()
    {
        return new WordDocument();
    }
}
public class WordXDocumentFactory : IDocumentFactory
{
    public IDocument Create()
    {
        return new WordXDocument();
    }
}

public class ExcelDocumentFactory : IDocumentFactory
{
    public IDocument Create()
    {
        return new ExcelDocument();
    }
}
public class ExcelXDocumentFactory : IDocumentFactory
{
    public IDocument Create()
    {
        return new ExcelXDocument();
    }
}
public class PowerPointDocumentFactory : IDocumentFactory
{
    public IDocument Create()
    {
        return new PowerPointDocument();
    }
}
public class PowerPointXDocumentFactory : IDocumentFactory
{
    public IDocument Create()
    {
        return new PowerPointXDocument();
    }
}

客戶端調用代碼:

static void Main(string[] args)
{
    IDocument document;
    IDocumentFactory documentFactory;

    documentFactory = new WordDocumentFactory();
    document = documentFactory.Create();
    document.Import();

    documentFactory = new WordXDocumentFactory();
    document = documentFactory.Create();
    document.Import();

    documentFactory = new ExcelDocumentFactory();
    document = documentFactory.Create();
    document.Import();

    documentFactory = new ExcelXDocumentFactory();
    document = documentFactory.Create();
    document.Import();

    documentFactory = new PowerPointDocumentFactory();
    document = documentFactory.Create();
    document.Import();

    documentFactory = new PowerPointXDocumentFactory();
    document = documentFactory.Create();
    document.Import();
   
    Console.ReadKey();
}

輸出結果:

image

 

一個產品一個實現類,一個工廠類,這樣職責單一符合SRP,但是系統中的類在成倍的增加,有點複雜了,如果在增加一個系列的產品那還了得。

那麼能不能減少一些類呢?經過分析我們發現,這些導入的文檔中2007之前的一系列文檔的解析規則基本類似實現的技術也是類似的,2007及以後的文檔的解析規則類似。所以我們可以把這些產品分成兩個系列,2007之前的成為Document系列,2007以後的文檔成為DocumentX系列, 那麼我們就可以創建兩個具體的工廠來創建Document系列和DocumentX系列, 從另一個維度來看,Word 和WordX, Excel 和ExcelX,PowerPoint 和 PowerPointX的的關係也很密切,因為都是同一個產品,只是處在不同的系列上,他們各自的編碼又各自類似,因此在這個維度上可以將其提出一組新的介面,IWordDocument 用於處理word的導入(Word和WordX),IExcelDocument 用處理Excel的導入(Excel 和ExcelX),IPowerPoint用於處理 PowerPoint 導入(PowerPoint和PowerPointX),根據這個思路重構代碼:

UML 圖

image

代碼:

public interface IWordDocument
{
    void Import();
}
public interface IExcelDocument
{
    void Import();
}
public interface IPowerPointDocument
{
    void Import();
}
public interface IDocumentFactory
{
    IWordDocument CreateWord();
    IExcelDocument CreateExcel();
    IPowerPointDocument CreatePowerPoint();
}
public class WordDocument : IWordDocument
{    
    public void Import()
    {
 	        Console.WriteLine("Import Word");
    }
}
public class WordXDocument : IWordDocument
{
    public void Import()
    {
        Console.WriteLine("Import WordX");
    }
}
public class ExcelDocument : IExcelDocument
{
    public void Import()
    {
 	    Console.WriteLine("Import Excel");
    }
}
public class ExcelXDocument : IExcelDocument
{
    public void Import()
    {
        Console.WriteLine("Import ExcelX");
    }
}
public class PowerPointDocument : IPowerPointDocument
{
    public void Import()
    {
        Console.WriteLine("Import Power Point");
    }
}
public class PowerPointXDocument : IPowerPointDocument
{
    public void Import()
    {
        Console.WriteLine("Import Power PointX");
    }
}
public class DocumentFactory : IDocumentFactory
{
    public IWordDocument CreateWord()
    {
        return new WordDocument();
    }

    public IExcelDocument CreateExcel()
    {
        return new ExcelDocument();
    }

    public IPowerPointDocument CreatePowerPoint()
    {
        return new PowerPointDocument();
    }
}
public class DocumentXFactory : IDocumentFactory
{
    public IWordDocument CreateWord()
    {
        return new WordXDocument();
    }

    public IExcelDocument CreateExcel()
    {
        return new ExcelXDocument();
    }

    public IPowerPointDocument CreatePowerPoint()
    {
        return new PowerPointXDocument();
    }
}

客戶端調用:

static void Main(string[] args)
{
    IWordDocument wordDocument;
    IExcelDocument excelDocument;
    IPowerPointDocument powerPointDocument;

    IDocumentFactory documentFactory;

    documentFactory = new DocumentFactory();
    wordDocument = documentFactory.CreateWord();
    excelDocument = documentFactory.CreateExcel();
    powerPointDocument = documentFactory.CreatePowerPoint();
    wordDocument.Import();
    excelDocument.Import();
    powerPointDocument.Import();

    documentFactory = new DocumentXFactory();
    wordDocument = documentFactory.CreateWord();
    excelDocument = documentFactory.CreateExcel();
    powerPointDocument = documentFactory.CreatePowerPoint();
    wordDocument.Import();
    excelDocument.Import();
    powerPointDocument.Import();    
   
    Console.ReadKey();
}

輸出:

image

這一次工廠的數量得到了控制,這裡只有兩個工廠類就搞定了,DocumentFactory 負責創建office 2007之前的文檔對象,DocumentXFactory 負責創建 office 2007以及以後的文檔對象。這就解決了工廠方法模式工廠類隨著產品增加隨之增加帶來的複雜性,以及不易維護的問題。這樣如果在增加一系列產品就變得容易了很多,只需要再創建一個系列產品的具體工廠並繼承自抽象工廠,以及實現系列產品的抽象介面的具體類就可以了。

隨著需求的變化一步一步的經過重構代碼也一步一步從簡單工廠模式到工廠方法模式再到抽象工廠模式了。現在文檔處理代碼就是一個典型的抽象工廠模式了,那麼下麵來看看抽象工廠模式到底是什麼?

二、抽象工廠模式定義:

抽象工廠模式(Abstract Factory Pattern):提供一個創建一系列相關或相互依賴對象的介面,而無須指定它們具體的類。抽象工廠模式又稱為Kit模式,它是一種對象創建型模式。在抽象工廠模式中,每一個具體工廠都提供了多個工廠方法用於產生多種不同類型的產品,這些產品構成了一個產品族(產品系列).

三、工廠方法模式結構圖:

image 抽象工廠模式結構圖

1、AbsctactFactory(抽象抽象工廠):

它聲明瞭一組用於創建一系列產品的方法,每一個方法對應創建一種產品。

2.ConcreteFactory(具體工廠):

它實現了在抽象工廠中聲明的創建產品的方法,生成一組具體產品,這些產品構成了一個產品系列,每個產品都在同一個系列中。

3. AbsctactProdcut(抽象產品):

它為每種產品聲明介面,在抽象產品中聲明瞭產品所具有的業務方法。

4.ConcreteProdcut(具體產品):

它定義具體工廠生產的具體產品對象,實現抽象產品介面中聲明的業務方法。

四、抽象工廠模式代碼實現

在抽象工廠中聲明瞭多個工廠方法,用於創建不同類型的產品,抽象工廠可以是介面,也可以是抽象類或者具體類,其典型代碼如下:

public interface IAbstractProdcutA
{
    void DoSomething();
}
public interface IAbstractProdcutB
{
    void DoSomething();
}
public interface IAbstractFactory{
    IAbstractProdcutA CreateProductA();
    IAbstractProdcutB CreateProductB();
}

public class ConcreteProductA1:IAbstractProdcutA
{
    public void DoSomething()
    {
 	    Console.WriteLine("I'm ConcreteProductA1");
    }
}
public class ConcreteProductA2:IAbstractProdcutA
{
    public void DoSomething()
    {
 	    Console.WriteLine("I'm ConcreteProductA2");
    }
}

public class ConcreteProductB1:IAbstractProdcutB
{
    public void DoSomething()
    {
 	    Console.WriteLine("I'm ConcreteProductB1");
    }
}
public class ConcreteProductB2:IAbstractProdcutB
{
    public void DoSomething()
    {
 	    Console.WriteLine("I'm ConcreteProductB2");
    }
}

public class ConcreteFactory1 : IAbstractFactory
{
    public IAbstractProdcutA CreateProductA()
    {
        return new ConcreteProductA1();
    }
    public IAbstractProdcutB CreateProductB()
    {
        return new ConcreteProductB1();
    }
}

public class ConcreteFactory2 : IAbstractFactory
{
    public IAbstractProdcutA CreateProductA()
    {
        return new ConcreteProductA2();
    }
    public IAbstractProdcutB CreateProductB()
    {
        return new ConcreteProductB2();
    }
}

客戶端調用代碼:

static void Main(string[] args)
{    
    IAbstractProdcutA productA;
    IAbstractProdcutB productB;

    IAbstractFactory factory = new ConcreteFactory1();
    productA = factory.CreateProductA();
    productB = factory.CreateProductB();
    productA.DoSomething();
    productB.DoSomething();

    factory = new ConcreteFactory2();
    productA = factory.CreateProductA();
    productB = factory.CreateProductB() ;
    productA.DoSomething();
    productB.DoSomething();
   
    Console.ReadKey();
}

輸出:

image

五 、抽象工廠模式的優點

  1. 抽象工廠模式隔離了具體類的生成,使得客戶並不需要知道什麼被創建。由於這種隔離,更換一個具體工廠就變得相對容易,所有的具體工廠都實現了抽象工廠中定義的那些公共介面,因此只需改變具體工廠的實例,就可以在某種程度上改變整個軟體系統的行為。比如常用的配置+反射就可輕鬆替換掉工廠進而替換掉整個以為邏輯。

  2. 當一個產品族中的多個對象被設計成一起工作時,它能夠保證客戶端始終只使用同一個產品系列中的對象。

  3. 增加新的產品系列很方便,無須修改已有系統,只需要增加具體產品類和具體工廠就可以了符合“開閉原則(OCP)“。

六、抽象工廠模式的缺點

  1. 增加新的產品麻煩,需要對原有系統進行較大的修改,甚至需要修改抽象層代碼,這顯然會帶來較大的不便,違背了“開閉原則(OCP)“。如果新增一個產品就要在抽象工廠中增加一個創建該產品的方法。這樣就牽一發動全身,原來的所有集成該抽象工廠的具體工廠都要修改。

七、抽象工廠模式的使用場景

  1. 一個系統不應當依賴於產品類實例如何被創建、組合和表達的細節,這對於所有類型的工廠模式都是很重要的,用戶無須關心對象的創建過程,將對象的創建和使用解耦。
  2. 系統中有多於一個的產品系列,而每次只使用其中某一產品系列。可以通過配置文件等方式來使得用戶可以動態改變產品系列,也可以很方便地增加新的產品系列。
  3. 屬於同一個產品系列的產品將在一起使用,這一約束必須在系統的設計中體現出來。同一個產品系列中的產品可以是沒有任何關係的對象,但是它們都具有一些共同的約束,如Office 文檔的支持,在window xp  重視不能使用Office 2007 +的,window 7 支持0ffice 2007,因此這個約束就是操作系統對office版本的支持。
  4. 產品類型穩定,設計完成之後,不會向系統中增加新的產品類型或者刪除已有的產品產品類型。

八、擴展

第三次需求改變:從商業化上考慮甲方要求這個軟體的某些版本只提供導入office 2007 之前的文件,某些版本只支持2007以後版本。

怎麼辦呢? 這個需求開發人員拿到後一定竊喜,抽象工廠實現這種需求簡直是太容易了。配置+反射呀, 將具體工廠配置在配置文件中,代碼中只是用抽象並通過反射來創建工廠實進而達到動態控制的能力。

如果軟體版本中V1中只要支持office 2007以前的文檔版本。那麼在配置文件App.config  增加以下節點:

image

配置完節點後在代碼中通過反射來創建工廠類, 代碼如下

static void Main(string[] args)
{
    IWordDocument wordDocument;
    IExcelDocument excelDocument;
    IPowerPointDocument powerPointDocument;

    IDocumentFactory documentFactory;

    // 讀取配置文件並且通過反射創建工廠實例
    var setting = ConfigurationSettings.AppSettings["DocumentFaccory"];
    var obj = Type.GetType(setting);
    if (obj == null) return;
    documentFactory = Activator.CreateInstance(obj) as IDocumentFactory;

    if (documentFactory == null) return;
 
    wordDocument = documentFactory.CreateWord();
    excelDocument = documentFactory.CreateExcel();
    powerPointDocument = documentFactory.CreatePowerPoint();
    wordDocument.Import();
    excelDocument.Import();
    powerPointDocument.Import();
   
    Console.ReadKey();
}

輸出結果如下:

image

如果在V1.1版本中甲方提出要求只支持office 2007以後的版本,那麼只需要修改配置文件如下就可以了:

把原來的

<appSettings>
    <add key="DocumentFaccory" value="DesignPattern.AbstractFactory.DocumentFactory"/>
</appSettings>

改成:

<appSettings>
    <add key="DocumentFaccory" value="DesignPattern.AbstractFactory.DocumentXFactory"/>
</appSettings>

其它地方不用做任何修改,輸出結果:

image


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 備註:我們經常會遇到使用jquery獲取某個地址下的部分頁面內容,然後替換當前頁面對應內容,也就是:局部刷新功能。 當然也可以使用get/post請求獲取數據,修改數據,可以參考以下JS代碼: ...
  • 一、安裝SubLime Text 3SubLime Text 3官網下載:https://www.sublimetext.com/ 二、安裝Package Control進入sublime後,按快捷鍵Ctrl+、鍵,進入命令行模式,sublime text 3在命令行輸入: sublime text ...
  • 一、React初探 es6寫法 "code" es5寫法(遺憾的是現在最新版本的react,已經不再能使用createClass去創建react組件了 "code" ) 核心思想:封裝組件,各個組件維護自己的狀態(state, prop)和UI,當狀態變更,自動重新渲染組件,數據流向是單向的。 需要 ...
  • 最近的幾個項目都用到了localStrage來存儲用戶的相關登錄信息,本來想繼續深入學習下localStrage的相關內容,逐漸衍生到了用戶登錄的相關邏輯 先整理下localStrage的相關知識點 1、http跟hppts不互通, 2、localStrage 存儲在 C:\Users\xxx\Ap ...
  • 看鬥魚有些時間了,也寫了不少輔助的js,但是昨天不小心把硬碟分區表搞沒了,自己寫了好久的代碼不見了,DiskGenius 也沒恢覆成功,所以要重寫一次,大家要引以為鑒,常備份代碼,github是個不錯的選擇。 先通過一個簡單的程式, 來認識下 Chrome 擴展,通過最簡單的2個文件,就能組成一個C ...
  • 空杯心態,從零開始。 應用運維->實操最重要。 每個業務至少有一種解決方案;每個方案至少會一個軟體實現。 系統架構師實際是軟體架構師。 系統運維架構師。 架構師主要是架構思想,不是某一個軟體個體。 製造子彈->系統開發級別。 組裝坦克飛機->鳳毛麟角。 Linux系統架構師->一期 軟體架構師->二 ...
  • RockBrain USB Server- 雲計算虛擬化USB設備集中管理、遠程共用解決方案(涉及銀企直聯) 技術需求: 1.企業員工的大量USB Key,需要將key接入USB Server虛擬池,進行集中管理。 2.設置USB Key最高管理員,本管理員許可權:在Server端可以對Key的埠進 ...
  • Date類: 類 Date 表示特定的瞬間,精確到毫秒。 毫秒概念:1000毫秒=1秒 毫秒的0點: System.currentTimeMillis() 返回值long類型參數 用於獲取當前日期的毫秒值 時間的原點:公元1970年 一月一日,午夜0:00:00 對應的毫秒值就是0 註意:時間和日期 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...