在 .NET中,一種更方便操作配置項的方法

来源:https://www.cnblogs.com/wpinfo/archive/2018/01/23/config_setting.html
-Advertisement-
Play Games

在應用程式的開發過程中,我們往往會為軟體提供一些配置項,以允許軟體根據配置項靈活來做事情,比如配置日誌文件路徑等,此外,我們還可以用配置項來為用戶存儲其偏好設置等。 .NET 為我們預設提供了配置機制以及配置文件,項目中的 app.config 或者 web.config 文件(如果沒有,可以添加) ...


在應用程式的開發過程中,我們往往會為軟體提供一些配置項,以允許軟體根據配置項靈活來做事情,比如配置日誌文件路徑等,此外,我們還可以用配置項來為用戶存儲其偏好設置等。

.NET 為我們預設提供了配置機制以及配置文件,項目中的 app.config 或者 web.config 文件(如果沒有,可以添加)就是 .NET 為我們提供的配置文件。在這個配置文件中的根節點 configuration 下,創建 appSettings 節點,在此節點中,我們可以添加自定義的配置項。同時,ConfigurationManager 類提供了訪問及操作此配置文件(由 Configuration 類代表)中配置的方法。需要註意的是,這個類在 System.Configuration.dll 文件中,需要將它添加到項目的引用中,才能使用。

本文主要介紹一種更為便利的方式來訪問/存儲配置項,當然,它本質上是使用 ConfigurationManager 類完成的。它主要的特點是以面向對象的方式來解決此問題,更具體地說,我們創建一個類,類中包括一些屬性用以表示配置項,通過訪問或設置這些屬性,即可得到或更新對應的配置項。

一、實現

首先,我們為 Configuration 類添加一個擴展方法 AddOrUpdateAppSettingItem,如下 :

        /// <summary>
        /// 向配置的 AppSetings 節添加(如果不存在)或更新(如果已存在)給定的 key 和 value
        /// </summary>
        /// <param name="config"></param>
        /// <param name="key"></param>
        /// <param name="value"></param>
        public static void AddOrUpdateAppSettingItem(this Configuration config, string key, string value)
        {
            if (config.AppSettings.Settings.AllKeys.Contains(key))
            {
                config.AppSettings.Settings[key].Value = value;
            }
            else
            {
                config.AppSettings.Settings.Add(key, value);
            }
        }
    }

這個方法主要實現了向 appSettings 節點添加(配置項不存在)或更新(配置項已存在)配置,接下來我們會用到這個方法。

接著,我們定義抽象基類 ConfigSetting,如下:

    public abstract class ConfigSetting
    {
        /// <summary>
        /// 配置類
        /// </summary>
        /// <param name="configuration">配置</param>
        public ConfigSetting(Configuration configuration)
        {
            Configuration = configuration;
        }

        /// <summary>
        /// 當前配置
        /// </summary>
        public Configuration Configuration
        {
            get;
        }

        /// <summary>
        /// 獲取當前程式配置
        /// </summary>
        /// <param name="config"></param>
        /// <returns></returns>
        public static Configuration GetCurrentConfiguration()
        {
            return ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None);
        }

        /// <summary>
        /// 返回指定的配置項的值
        /// </summary>
        /// <param name="settingKey"></param>
        /// <returns></returns>
        protected virtual string GetSettingValue([CallerMemberName] string settingKey = null)
        {
            return Configuration?.AppSettings?.Settings[settingKey]?.Value;
        }

        /// <summary>
        /// 返回指定的配置項的值
        /// </summary>
        /// <typeparam name="T">值類型</typeparam>
        /// <param name="settingKey"></param>
        /// <returns></returns>
        protected virtual T GetSettingValue<T>([CallerMemberName] string settingKey = null)
        {
            var value = GetSettingValue(settingKey);

            if (string.IsNullOrWhiteSpace(value))
            {
                return default(T);
            }
            else
            {
                return (T)Convert.ChangeType(value, typeof(T));
            }
        }

        /// <summary>
        /// 為指定的配置項設置值
        /// </summary>
        /// <param name="value"></param>
        /// <param name="settingKey"></param>
        protected virtual void SetSettingValue(object value, [CallerMemberName] string settingKey = null)
        {
            Configuration.AddOrUpdateAppSettingItem(settingKey, value?.ToString());
            Configuration.Save();
        }
    }

其中主要包括了一個靜態方法和三個受保護的 virtual 方法,說明:

1. 靜態方法 GetCurrentConfiguration 返回當前應用的配置類;
2. GetSettingValue 和 SetSettingValue 方法則分別負責讀取、設置指定配置項的值;
3. GetSettingValue 有兩個重載,其中一個用於支持泛型;
4. 在它們的方法簽名中包括 CallerMemberName 特性,通過這個屬性可以得到調用此訪問的方法或屬性的名稱。

然後,創建一個名為 AppConfigSetting 的類,這個類將會包括一些代表配置項的屬性,並且它要繼承自 ConfigSetting,如下:

    public class AppConfigSetting : ConfigSetting
    {
        public AppConfigSetting(Configuration configuration) : base(configuration)
        {
            
        }

        public DateTime InstallDateTime
        {
            get { return GetSettingValue<DateTime>(); }
            set { SetSettingValue(value); }
        }

        public string LogFileName
        {
            get { return GetSettingValue(); }
            set { SetSettingValue(value); }
        }

        public int ReadBlockSize
        {
            get { return GetSettingValue<int>(); }
            set { SetSettingValue(value); }
        }
    }

說明:

1. 可以看到我們在其中增加了三個屬性。而在它們的 get 與 set 段中,調用了基類中的對應的兩個方法,其中,對於非 string 類型的配置項,我們調用的是 GetSettingValue<T>。
2. 通過我們前面提到的 CallerMemberName 特性,就可以得到這裡的屬性名,並得到相應的配置項,這樣我們就無需硬編碼。所以,這個屬性名,本質上就是配置項的名稱。

這樣,我們將所有要用到的配置項作為屬性放到 AppConfigSetting 類中,再用操作這些屬性就可以完成所有對配置項的操作了。 

二、如何使用

使用它,也非常簡單,代碼如下: 

            var config = ConfigSetting.GetCurrentConfiguration();
            AppConfigSetting setting = new AppConfigSetting(config);
            
            // 未設置時
            MessageBox.Show($"LogFileName: {setting.LogFileName}");

            // 設置後,再讀取
            setting.LogFileName = "log.txt";
            MessageBox.Show($"LogFileName: {setting.LogFileName}");

 三、補充

為了滿足在不向 AppConfigSetting 添加配置項屬性,但卻又要訪問/存儲指定配置項的需要,我們可以在基類 ConfigSetting 中添加以下三個方法:

        /// <summary>
        /// 返回指定的配置項的值
        /// </summary>
        /// <param name="settingKey"></param>
        /// <returns></returns>
        public string GetSettingValueByKey(string settingKey)
        {
            return GetSettingValue(settingKey);
        }

        /// <summary>
        /// 返回指定的配置項的值
        /// </summary>
        /// <param name="settingKey"></param>
        /// <returns></returns>
        public T GetSettingValueByKey<T>(string settingKey)
        {
            return GetSettingValue<T>(settingKey);
        }

        /// <summary>
        /// 為指定的配置項設置值
        /// </summary>
        /// <param name="value"></param>
        /// <param name="settingKey"></param>
        public void SetSettingValueByKey(string settingKey, object value)
        {
            SetSettingValue(value, settingKey);
        }

使用這幾個方法,可以自由地訪問/存儲配置項,不同於上面增加屬性方式的是,它需要自己傳遞配置項 key 作參數。

使用:

            // 未設置時
            MessageBox.Show($"LogLevel: {setting.GetSettingValueByKey("LogLevel")}");

            // 設置後,再讀取
            setting.SetSettingValueByKey("LogLevel", 5);
            MessageBox.Show($"LogLevel: {setting.GetSettingValueByKey("LogLevel")}");

 總結

本文主要介紹了一種訪問應用程式配置更為便利的方式,其主要思想是通過基類中 GetSettingValue/SetSettingValue 兩個方法藉助於 CallerMemberName 特性而得到派生類中屬性的名稱而操作對應的配置項。當然,這裡不僅提供了一種方法,更是提供一種思路,基於此,你還可以根據實際需要來調整、擴展以滿足你的實際需要。 

源碼下載


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

-Advertisement-
Play Games
更多相關文章
  • 前言 redis 大家都使用過, 可以安裝在windows下, 也可以安裝在linux下, 一般還是linux下安裝比較多. 這裡來介紹一下redis在linux下的安裝 一. 下載 https://redis.io/ 二. 安裝 1. 下載之後, 將文件拷貝到 linux 中. 我這裡是放在 /h ...
  • ###反射## getattr,hasattr,setattr,delattr,和類裡面的欄位有關,具體看例子 #1 class Person: def __init__(self,name,age): self.name = name self.age = age def show_lover(s ...
  • java.util.Date 就是在除了SQL語句的情況下麵使用 java.sql.Date 是針對SQL語句使用的,它只包含日期而沒有時間部分 直接說就是:java.sql.Date就是與資料庫Date相對應的一個類型,而java.util.Date是純java的Date 它都有getTime方法 ...
  • 1:列表: Python的列表比C語言的數組強大的多,數組只能存放相同類型的數據,而列表則像一個大集裝箱可以存放整形、浮點型、字元串、對象等 2:創建列表的方法 3:向列表中添加元素的方法 1)append() 向列表末尾添加一個參數 2)extend() 參數為一個列表,從原列表擴展原有列表 3) ...
  • 在 《不一樣的Interceptor》中我們著重介紹了Dora.Interception中最為核心的對象Interceptor,以及定義Interceptor類型的一些約定。由於Interceptor總是通過攔截某個方法的調用進而實現對前置或者後置操作的註入,所以我們定義的Interceptor類型... ...
  • 上一篇文章介紹了OAuth2.0以及如何使用.Net來實現基於OAuth的身份驗證,本文是對上一篇文章的補充,主要是介紹OAuth與Jwt以及OpenID Connect之間的關係與區別。 本文主要內容有: ● Jwt簡介 ● .Net的Jwt實現 ● OAuth與Jwt ● .Net中使用Jwt ...
  • 1. 前言 之前在 "如何使用Fluent Design System" 這篇文章里已經簡單介紹過Reveal的用法,這篇再詳細介紹其它內容。 2. 自定義RevealButtonStyle 我覺得常用ItemsControl都已經自動應用了Reveal,用就是了。 沒有預設應用Reveal的控制項, ...
  • 最近項目中需要頻繁用到伺服器返回的Cookie,由於項目採用的是HttpClient,並且用CookieContainer自動托管Cookie,在獲取Cookie的時候不太方便。所以就寫了個拓展類。 首先的遍歷獲取所有的Cookie,然後存入字典當中,再把字典存入記憶體緩存中,在下次調用方法的時候,先 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...