C#開發微信門戶及應用(42)--使用Autofac實現微信介面處理的控制反轉處理

来源:http://www.cnblogs.com/wuhuacong/archive/2016/06/17/5593361.html
-Advertisement-
Play Games

在很多情況下,我們利用IOC控制反轉可以很方便實現一些介面的適配處理,可以在需要的時候切換不同的介面實現,使用這種方式在調用的時候,只需要知道相應的介面介面,具體調用哪個實現類,可以在配置文件中動態指定,本篇主要介紹AutoFac的IOC組件的使用,用來實現微信介面處理的控制反轉功能。 ...


在很多情況下,我們利用IOC控制反轉可以很方便實現一些介面的適配處理,可以在需要的時候切換不同的介面實現,使用這種方式在調用的時候,只需要知道相應的介面介面,具體調用哪個實現類,可以在配置文件中動態指定,本篇主要介紹AutoFac的IOC組件的使用,用來實現微信介面處理的控制反轉功能。

我們知道,實現IOC的方式有很多,如Unity、AutoFac、Ninject、Castle Windsor、Spring.NET等等,每種IOC組件均有自己的一些特點,我在之前的實體框架隨筆系列介紹過Unity的使用《Entity Framework 實體框架的形成之旅--利用Unity對象依賴註入優化實體框架(2)》,本來也想用這個來實現微信的介面調用處理,不過由於其版本以及一些其他問題,總是沒有那麼方便,最後決定使用也比較流行,應用較多的的AutoFac組件來實現。

1、微信介面的處理需求

我們在使用微信公眾號實現一些業務處理的時候,往往需要根據不同的條件進行不同的介面調用。

如通過二維碼掃碼的結果處理,然後呈現給微信用戶的相關信息,有下麵兩種方式。

根據用戶的掃碼結果,我們可以自定義自己的業務處理,然後呈現給用戶,那麼這裡使用IOC來實現具體的業務是比較好的,我們在具體的業務實現裡面,可以根據不同的條件實現所需要的複雜處理。

當然我們還可以擴展到很多的業務介面裡面,如百度的地理位置解析介面、電影院信息查詢、天氣信息查詢、交通信息查詢、旅游信息查詢等,還有簡訊、郵件發送等常規介面,都可以使用這種方式進行處理。

介面的效果展示如下所示。

這些給其他項目模塊使用的時候,我們可以在配置文件裡面指定具體的介面實現信息,這種可以具體指定所需的實現。

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <configSections>
    <section name="autofac" type="Autofac.Configuration.SectionHandler, Autofac.Configuration"/>
  </configSections>
  <autofac defaultAssembly="WHC.Common.Handler">
    <components>
      <component type="WHC.Common.Handler.TestHandler, WHC.Common.Handler" service="WHC.Common.Handler.ITestHandler" />
      <component type="WHC.Common.Handler.QRCodeHandler, WHC.Common.Handler" service="WHC.Common.Handler.IQRCodeHandler" />
      <!--郵件簡訊-->
      <component type="WHC.Common.Handler.SmsSendHandler, WHC.Common.Handler" service="WHC.Common.Handler.ISmsHandler" />
      <component type="WHC.Common.Handler.MailSendHandler, WHC.Common.Handler" service="WHC.Common.Handler.IMailHandler" />
      
    </components>
  </autofac>
</configuration>

直接使用AutoFac的操作應該是比較方便,使用介面獲取方式獲取具體實現就可以了。

 

2、使用Autofac實現

為了方便使用Autofac,我們可以先在項目上的Nuget包管理,引用相關的DLL,其中包括核心的Autofac類庫,以及讀取配置文件的Autofac Configuration,後者為方便讀取XML配置信息所必須。

引入這兩個DLL就可以使用Autofac的功能了。

一般通過配置文件,初始化的Autofac組件的代碼如下所示

    instance = new AutoFactory();

    //初始化相關的註冊介面
    var builder = new ContainerBuilder();
    //從配置文件註冊相關的介面處理
    builder.RegisterModule(new ConfigurationSettingsReader("autofac", configurationFile));
    container = builder.Build();

而我們使用Autofac的介面也是很容易的,常規的使用代碼如下所示。

            var handler = container.Resolve<ITestHandler>();
            handler.Test("測試");

當然,為了方便,我們可以使用一個輔助類來簡化這個介面的調用:在輔助類初始化的時候,我們從配置文件載入對應的組件介面實現,當我們需要解析具體介面的時候,就可以直接從Container容器裡面胡獲取了,輔助類代碼如下所示。

    /// <summary>
    /// 使用AutoFac的工廠類,通過配置
    /// </summary>
    public class AutoFactory
    {
        //普通局部變數
        private static object syncRoot = new Object();
        //工廠類的單例
        private static AutoFactory instance = null;
        //配置文件
        private const string configurationFile = "autofac.config";

        /// <summary>
        /// IOC的容器,可調用來獲取對應介面實例。
        /// </summary>
        public IContainer Container { get; set; }

        /// <summary>
        /// IOC容器工廠類的單例
        /// </summary>
        public static AutoFactory Instatnce
        {
            get
            {
                if (instance == null)
                {
                    lock (syncRoot)
                    {
                        if (instance == null)
                        {
                            instance = new AutoFactory();

                            //初始化相關的註冊介面
                            var builder = new ContainerBuilder();
                            //從配置文件註冊相關的介面處理
                            builder.RegisterModule(new ConfigurationSettingsReader("autofac", configurationFile));
                            instance.Container = builder.Build();
                        }
                    }
                }
                return instance;
            }
        }

        /// <summary>
        /// 測試的介面
        /// </summary>
        public void Test()
        {
            var handler = AutoFactory.Instatnce.Container.Resolve<ITestHandler>();
            handler.Test("測試");
        }
    }

 

3、外部介面實現及調用

這樣我們所有的介面都定義好,並給每個定義的介面相應個實現就可以使用這個Autofac組件進行調用了。

    /// <summary>
    /// 簡訊發送介面
    /// </summary>
    public interface ISmsHandler
    {               
        /// <summary>
        /// 發送簡訊
        /// </summary>
        /// <param name="content">簡訊內容</param>
        /// <param name="mobiles">手機號碼(多個號碼用”,”分隔)</param>
        /// <param name="sendTime">預約發送時間</param>
        /// <returns></returns>
        CommonResult Send(string content, string mobiles, DateTime? sendTime = null);
                        
        /// <summary>
        /// 查詢剩餘條數
        /// </summary>
        /// <returns></returns>
        CommonResult GetLeftCount();
    }
    /// <summary>
    /// 郵件發送介面
    /// </summary>
    public interface IMailHandler
    {              
        /// <summary>
        /// 發送外部郵件(自定義郵件配置,如個人郵件)
        /// </summary>
        /// <param name="mailInfo">發送郵件信息</param>
        /// <param name="settingInfo">SMTP協議設置信息</param>
        /// <returns></returns>
        CommonResult Send(MailInfo mailInfo, SmtpSettingInfo settingInfo);
                        
        /// <summary>
        /// 發送外部郵件(系統配置,系統郵件)
        /// </summary>
        /// <param name="mailInfo">發送郵件信息</param>
        /// <returns></returns>
        CommonResult Send(MailInfo mailInfo);
    }

例如,測試發送簡訊和郵件的IOC調用代碼如下所示

            //使用IOC模塊發送
            var sms = AutoFactory.Instatnce.Container.Resolve<ISmsHandler>();
            var smsTemplate = string.Format("驗證碼:{0}。尊敬的會員,您好,您正在註冊會員,驗證碼2分鐘內有效,感謝您的支持。", new Random().Next(100000));
            var result = sms.Send(smsTemplate, "18620292076");
            Console.WriteLine(result.Success ? "發送簡訊成功" : "發送簡訊失敗:" + result.ErrorMessage);

            MailInfo info = new MailInfo();
            info.ToEmail = "[email protected]";
            info.FromEmail = "[email protected]";
            info.Subject = "這是一份來自我自己的測試郵件";
            info.Body = info.Subject + ",這是內容部分。<a href='http://www.iqidi.com'>點擊這裡返回主頁</a>";
            var mail = AutoFactory.Instatnce.Container.Resolve<IMailHandler>();

            var mailResult = mail.Send(info);
            Console.WriteLine(mailResult.Success ? "發送郵件成功" : "發送郵件失敗:" + mailResult.ErrorMessage);

測試後得到的結果如下:

郵件結果一樣可以收到。

我們回到上面介紹的二維碼掃描的業務實現效果,上面提到了,一個二維碼事件可以派生出不同的介面實現,從而給不同的響應信息。

    /// <summary>
    /// 掃碼進行的處理
    /// </summary>
    public interface IQRCodeHandler
    {
        /// <summary>
        /// 處理ScancodePush的事件
        /// </summary>
        /// <param name="info">掃描信息</param>
        /// <param name="accountInfo">賬號信息</param>
        /// <returns></returns>
        string HandleScancodePush(RequestEventScancodePush info, AccountInfo accountInfo);

        /// <summary>
        /// 處理ScancodeWaitmsg的事件
        /// </summary>
        /// <param name="info">掃描信息</param>
        /// <param name="accountInfo">賬號信息</param>
        /// <returns></returns>
        string HandleScancodeWaitmsg(RequestEventScancodeWaitmsg info, AccountInfo accountInfo);
    }

我們可以定義兩個簡單的介面處理,用來承接微信二維碼掃描介面的處理操作。

這樣我們在處理二維碼掃描事件的時候,我們就可以把它分配到介面裡面進行處理即可。

        /// <summary>
        /// 掃碼推事件的事件推送處理
        /// </summary>
        /// <param name="info">掃描信息</param>
        /// <returns></returns>
        public string HandleEventScancodePush(RequestEventScancodePush info, AccountInfo accountInfo)
        {
            string result = "";
            var handler = AutoFactory.Instatnce.Container.Resolve<IQRCodeHandler>();
            if(handler != null)
            {
                result = handler.HandleScancodePush(info, accountInfo);
            }
            return result;
        }

        /// <summary>
        /// 掃碼推事件且彈出“消息接收中”提示框的事件推送的處理
        /// </summary>
        /// <param name="info">掃描信息</param>
        /// <returns></returns>
        public string HandleEventScancodeWaitmsg(RequestEventScancodeWaitmsg info, AccountInfo accountInfo)
        {
            string result = "";
            try
            {
                var handler = AutoFactory.Instatnce.Container.Resolve<IQRCodeHandler>();
                if (handler != null)
                {
                    result = handler.HandleScancodeWaitmsg(info, accountInfo);
                }
            }
            catch(Exception ex)
            {
                LogHelper.Error(ex);
            }
            return result;
        }

對於其中之一的介面處理,我們都可以把它分拆,根據掃描的事件鍵值Key進行不同的信息相應。

        /// <summary>
        /// 掃描後,會等待事件處理結果返回給用戶
        /// </summary>
        public string HandleScancodeWaitmsg(RequestEventScancodeWaitmsg info, AccountInfo accountInfo)
        {
            ResponseText response = new ResponseText(info);
            response.Content = string.Format("您的信息為:{0},可以結合後臺進行數據查詢。", info.ScanCodeInfo.ScanResult);
            var result = response.ToXml();

            string devicecode = GetParam(info.ScanCodeInfo, "devicecode");//參數名為小寫
            if (!string.IsNullOrEmpty(devicecode))
            {
                switch(info.EventKey.ToLower())
                {
                    case "device_view"://設備查看
                        {
                            var deviceinfo = BLLFactory<Device>.Instance.FindByCode(devicecode);
                            response.Content = ConvertDeviceInfo(deviceinfo);
                            result = response.ToXml();
                        }
                        break;

                    case "measure"://設備計量
                        {
                            var deviceinfo = BLLFactory<Device>.Instance.FindByCode(devicecode);
                            response.Content = ConvertMeasure(deviceinfo);
                            result = response.ToXml();
                        }
                        break;

                    case "repair"://設備報修,返回報修單號
                        {
                            var content = ConvertRepaire(info, accountInfo, devicecode);
                            response.Content = content;
                            result = response.ToXml();
                        }
                        break;

                    case "inventory"://設備盤點,轉到盤點界面
                        {
                            var content = ConvertInventory(info, accountInfo, devicecode);
                            response.Content = content;
                            result = response.ToXml();
                        }
                        break;

                    case "maintain":
                        break;

                    case "check":
                        break;
                    case "device_add":
                        break;
                }
            }

            return result;
        }

以上就是關於使用Autofac實現一些常規介面處理的實現,這種控制反轉的方式,可以便於我們項目的開發效率,可以根據需要指定一些特定的實現處理即可,而且通過配置文件的方式載入,可以很方便的進行配置。

 


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

-Advertisement-
Play Games
更多相關文章
  • 在開發ASP.NET MVC時,我們會遇上這樣的情形,需要一次性傳送多個Model從控制器Controller至視圖View。 實現很簡單,只是創建一個集合類即可。Ok,下麵先在資料庫準備一些數據,如: CREATE TABLE [dbo].[TableA] ( [A] NVARCHAR(30) N ...
  • 前言 如果你還不知道ZKEACMS,不妨先瞭解一下。 ASP.NET MVC 開源建站系統 ZKEACMS 推薦,從此網站“拼”起來 官方地址:http://www.zkea.net/zkeacms 下載地址:https://github.com/SeriaWei/ASP.NET-MVC-CMS/r ...
  • 原文: "Working with SQL Server LocalDB" 作者: "Rick Anderson" 翻譯: "魏美娟(初見)" 校對: "孟帥洋(書緣)" 、 "張碩(Apple)" 、 "許登洋(Seay)" 類負責連接資料庫並將 對象和數據記錄進行映射。 Startup.cs 文 ...
  • 前言 GPS測量儀測量的產地面積,然後提交到系統中,系統需要校驗這塊產地和其他產地是否有重疊,重疊超過10%就要提出警告這塊產地已經被XXX登記入庫了。GPS測量儀測量出來的數據是連續的經緯度坐標數據。現在的問題就轉換成求一個一系列點圍成的區域和其他區域是否存在交集。拿到這個需求我想應該很簡單,網上 ...
  • 通過批處理操作註冊表實現winform應用中Webbrowser以指定的IE版本載入網頁 參考博文:http://www.cnblogs.com/liuzhendong/archive/2012/03/21/2410107.html ...... ...
  • 按[Ctrl+Shift+F]彈出查找視窗(不統計以#號開頭、以/開頭的代碼和空行) 1、輸入 :b*[^:b#/]+.*$ 2、選擇使用正則表達式 3、查找文件類型,*.cs多種類型用分號(;)隔開 點擊查找全部(查找結果如下) ...
  • 在WPF中 控制項綁定數據源時,數據源建議採用 ObservableCollection<T>集合 ObservableCollection<T> 類:表示一個動態數據集合,在添加項、移除項或刷新整個列表時,此集合將動態提供通知。數據源發生更改時,控制項內容也自動發生相應修改。無需重新綁定。 list< ...
  • 在上一篇博客中,筆者分享了一些從頁面整體的角度對頁面與ViewModel的思考。在本文中筆者希望從相對細節的角度分享一些對頁面與ViewModel的思考。 比如,當我們在更新View Model中的綁定數據時,應該怎樣更新呢?簡單的自然可以用新的數據實例直接替代舊的,但是這樣容易造成UI界面閃爍。尤 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...