.NET Attribute在數據校驗上的應用

来源:https://www.cnblogs.com/hexu0512/archive/2020/05/12/12879671.html
-Advertisement-
Play Games

Attribute(特性)的概念不在此贅述了,相信有點.NET基礎的開發人員都明白,用過Attribute的人也不在少數,畢竟很多框架都提供自定義的屬性,類似於Newtonsoft.JSON中JsonProperty、JsonIgnore等 自定義特性 .NET 框架允許創建自定義特性,用於存儲聲明 ...


Attribute(特性)的概念不在此贅述了,相信有點.NET基礎的開發人員都明白,用過Attribute的人也不在少數,畢竟很多框架都提供自定義的屬性,類似於Newtonsoft.JSON中JsonProperty、JsonIgnore等

自定義特性

.NET 框架允許創建自定義特性,用於存儲聲明性的信息,且可在運行時被檢索。該信息根據設計標準和應用程式需要,可與任何目標元素相關。

創建並使用自定義特性包含四個步驟:

  • 聲明自定義特性
  • 構建自定義特性
  • 在目標程式元素上應用自定義特性
  • 通過反射訪問特性

聲明自定義特性

一個新的自定義特性必須派生自System.Attribute類,例如:

public class FieldDescriptionAttribute : Attribute
{
    public string Description { get; private set; }

    public FieldDescriptionAttribute(string description)
    {
        Description = description;
    }
}
public class UserEntity
{
    [FieldDescription("用戶名稱")]
    public string Name { get; set; }
}

該如何拿到我們標註的信息呢?這時候需要使用反射獲取

      var type = typeof(UserEntity);
      var properties = type.GetProperties();
      foreach (var item in properties)
      {
          if(item.IsDefined(typeof(FieldDescriptionAttribute), true))
          {
              var attribute = item.GetCustomAttribute(typeof(FieldDescriptionAttribute)) as FieldDescriptionAttribute;
              Console.WriteLine(attribute.Description);
          }
      }

執行結果如下:

從執行結果上看,我們拿到了我們想要的數據,那麼這個特性在實際使用過程中,到底有什麼用途呢?

Attribute特性妙用

在實際開發過程中,我們的系統總會提供各種各樣的對外介面,其中參數的校驗是必不可少的一個環節。然而沒有特性時,校驗的代碼是這樣的:

  public class UserEntity
  {
      /// <summary>
      /// 姓名
      /// </summary>
      [FieldDescription("用戶名稱")]
      public string Name { get; set; }

      /// <summary>
      /// 年齡
      /// </summary>
      public int Age { get; set; }

      /// <summary>
      /// 地址
      /// </summary>
      public string Address { get; set; }
  }
      UserEntity user = new UserEntity();

      if (string.IsNullOrWhiteSpace(user.Name))
      {
          throw new Exception("姓名不能為空");
      }
      if (user.Age <= 0)
      {
          throw new Exception("年齡不合法");
      }
      if (string.IsNullOrWhiteSpace(user.Address))
      {
          throw new Exception("地址不能為空");
      }

欄位多了之後這種代碼就看著非常繁瑣,並且看上去不直觀。對於這種繁瑣又噁心的代碼,有什麼方法可以優化呢?
使用特性後的驗證寫法如下:

首先定義一個基礎的校驗屬性,提供基礎的校驗方法

    public abstract class AbstractCustomAttribute : Attribute
    {
        /// <summary>
        /// 校驗後的錯誤信息
        /// </summary>
        public string ErrorMessage { get; set; }

        /// <summary>
        /// 數據校驗
        /// </summary>
        /// <param name="value"></param>
        public abstract void Validate(object value);
    }

然後可以定義常用的一些對應的校驗Attribute,例如RequiredAttribute、StringLengthAttribute

        /// <summary>
        /// 非空校驗
        /// </summary>
        [AttributeUsage(AttributeTargets.Property)]
        public class RequiredAttribute : AbstractCustomAttribute
        {
            public override void Validate(object value)
            {
                if (value == null || string.IsNullOrWhiteSpace(value.ToString()))
                {
                    throw new Exception(string.IsNullOrWhiteSpace(ErrorMessage) ? "欄位不能為空" : ErrorMessage);
                }
            }
        }

        /// <summary>
        /// 自定義驗證,驗證字元長度
        /// </summary>
        [AttributeUsage(AttributeTargets.Property)]
        public class StringLengthAttribute : AbstractCustomAttribute
        {
            private int _maxLength;
            private int _minLength;

            public StringLengthAttribute(int minLength, int maxLength)
            {
                this._maxLength = maxLength;
                this._minLength = minLength;
            }

            public override void Validate(object value)
            {
                if (value != null && value.ToString().Length >= _minLength && value.ToString().Length <= _maxLength)
                {
                    return;
                }

                throw new Exception(string.IsNullOrWhiteSpace(ErrorMessage) ? $"欄位長度必須在{_minLength}與{_maxLength}之間" : ErrorMessage);
            }
        }

添加一個用於校驗的ValidateExtensions

public static class ValidateExtensions
  {
      /// <summary>
      /// 校驗
      /// </summary>
      /// <typeparam name="T"></typeparam>
      /// <returns></returns>
      public static void Validate<T>(this T entity) where T : class
      {
          Type type = entity.GetType();

          foreach (var item in type.GetProperties())
          {
              //需要對Property的欄位類型做區分處理針對Object List 數組需要做區分處理
              if (item.PropertyType.IsPrimitive || item.PropertyType.IsEnum || item.PropertyType.IsValueType || item.PropertyType == typeof(string))
              {
                  //如果是基元類型、枚舉類型、值類型或者字元串 直接進行校驗
                  CheckProperty(entity, item);
              }
              else
              {
                  //如果是引用類型
                  var value = item.GetValue(entity, null);
                  CheckProperty(entity, item);
                  if (value != null)
                  {
                      if ((item.PropertyType.IsGenericType && Array.Exists(item.PropertyType.GetInterfaces(), t => t.GetGenericTypeDefinition() == typeof(IList<>))) || item.PropertyType.IsArray)
                      {
                          //判斷IEnumerable
                          var enumeratorMI = item.PropertyType.GetMethod("GetEnumerator");
                          var enumerator = enumeratorMI.Invoke(value, null);
                          var moveNextMI = enumerator.GetType().GetMethod("MoveNext");
                          var currentMI = enumerator.GetType().GetProperty("Current");
                          int index = 0;
                          while (Convert.ToBoolean(moveNextMI.Invoke(enumerator, null)))
                          {
                              var currentElement = currentMI.GetValue(enumerator, null);
                              if (currentElement != null)
                              {
                                  currentElement.Validate();
                              }
                              index++;
                          }
                      }
                      else
                      {
                          value.Validate();
                      }
                  }
              }
          }
      }

      private static void CheckProperty(object entity, PropertyInfo property)
      {
          if (property.IsDefined(typeof(AbstractCustomAttribute), true))//此處是重點
          {
              //此處是重點
              foreach (AbstractCustomAttribute attribute in property.GetCustomAttributes(typeof(AbstractCustomAttribute), true))
              {
                  if (attribute == null)
                  {
                      throw new Exception("AbstractCustomAttribute not instantiate");
                  }

                  attribute.Validate(property.GetValue(entity, null));
              }
          }
      }
  }

新的實體類

  public class UserEntity
  {
      /// <summary>
      /// 姓名
      /// </summary>
      [Required]
      public string Name { get; set; }

      /// <summary>
      /// 年齡
      /// </summary>
      public int Age { get; set; }

      /// <summary>
      /// 地址
      /// </summary>
      [Required]
      public string Address { get; set; }

      [StringLength(11, 11)]
      public string PhoneNum { get; set; }
  }

調用方式

UserEntity user = new UserEntity();
user.Validate();

上面的校驗邏輯寫的比較複雜,主要是考慮到對象中包含複雜對象的情況,如果都是簡單對象,可以不用考慮,只需針對單個屬性做欄位校驗
現有的方式是在校驗不通過的時候拋出異常,此處大家也可以自定義異常來表示校驗的問題,也可以返回自定義的校驗結果實體來記錄當前是哪個欄位出的問題,留待大家自己實現
如果您有更好的建議和想法歡迎提出,共同進步

以上代碼均為原創分享,若大家認為有不妥的地方,煩請留言指出,在下感激不盡

本文作者:hexuwsbg

出處:https://www.cnblogs.com/hexu0512/p/12879671.html

版權:本文采用「可附帶出處轉載」知識共用許可協議進行許可


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

-Advertisement-
Play Games
更多相關文章
  • 1 ## 類 2 ''' 3 在面向對象編程中,你編寫表示現實世界中的事物和情景的類,並基於這些類來創建對象。 4 編寫類時,你定義一大類對象都有的通用行為。基於類創建 對象時,每個對象都自動具備 5 這種通用行為,然後可根據需要賦予每個對象獨特的個性。 6 7 根據類來創建對象被稱為 實例化,這讓 ...
  • 我們知道從 PHP 5.3 起三元運算符 ? : 有一個寫法簡潔寫法是這樣的: <?php $a = 0; $b = $a ?: 1; # $b 1 這實際上相當於: <?php $a = 0; $b = $a ? $a : 1; # $b 1 在 PHP5 中,語法分析是這樣寫的: | expr ...
  • Go 語言是一種靜態類型的編程語言。這意味著,編譯器需要在編譯時知曉程式里每個值的類型。數據類型的出現是為了把數據分成所需記憶體大小不同的數據,編程的時候需要用大數據的時候才需要申請大記憶體,就可以充分利用記憶體。 Go語言內置以下這些基礎類型: 布爾類型:bool 整型:int8、byte、int16、 ...
  • 1、首先準備好數據。這裡的數據不是直接從資料庫中查到的數據而是將查到的數據複製一份,兩者的數據互不影響,這樣有利於複製之後的數據可以修改。 ① 定義一個從資料庫中查到的數據的方法(service層的實現類方法),這裡省略mapper映射文件和dao層介面的方法 /** * 參數是實體類,根據參數條件 ...
  • 安裝docker 1.更新yum包 yum update 2.卸載舊版本 yum remove docker 3.安裝依賴 yum install -y yum-utils device-mapper-persistent-data lvm2 4.設置yum源 yum-config-manager ...
  • 本博客系列是學習併發編程過程中的記錄總結。由於文章比較多,寫的時間也比較散,所以我整理了個目錄貼(傳送門),方便查閱。 "併發編程系列博客傳送門" 本文是轉載問斬,原文請見 "這裡" 一、Exchanger簡介 Exchanger——交換器,是JDK1.5時引入的一個同步器,從字面上就可以看出,這個 ...
  • 1.Xpath Helper Xpath Helper 是一個面向 Xpath 初學者的 Google Chrome 插件。相對於人工找 Xpath 語法,Xpath Helper 可以實現自動分析。只要你打開一個網頁,然後點擊任何一個網路元素,Xpath Helper 就能自動幫你找出相應的 Xp ...
  • 工作的時候,尤其是自媒體,我們必備水印添加工具以保護我們的知識產權,網上有許多的線上 / 下載的水印添加工具,但他們或多或少都存在以下問題: 線上工具需要上傳到對方伺服器,信息不安全。 很多工具不具備批量處理功能。 很多工具自定義的功能太少,如水印透明度,字體等。 操作繁瑣。這裡還要註意:光理論是不 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...