溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C#泛型設(shè)計(jì)需要注意什么

發(fā)布時(shí)間:2021-08-19 14:48:59 來源:億速云 閱讀:109 作者:小新 欄目:編程語言

這篇文章主要為大家展示了“C#泛型設(shè)計(jì)需要注意什么”,內(nèi)容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“C#泛型設(shè)計(jì)需要注意什么”這篇文章吧。

什么是泛型

我們?cè)诰帉懗绦驎r(shí),經(jīng)常遇到兩個(gè)模塊的功能非常相似,只是一個(gè)是處理int數(shù)據(jù),另一個(gè)是處理string數(shù)據(jù),或者其他自定義的數(shù)據(jù)類型,但我們沒有辦法,只能分別寫多個(gè)方法處理每個(gè)數(shù)據(jù)類型,因?yàn)榉椒ǖ膮?shù)類型不同。有沒有一種辦法,在方法中傳入通用的數(shù)據(jù)類型,這樣不就可以合并代碼了嗎?泛型的出現(xiàn)就是專門解決這個(gè)問題的。

背景

最近一直在對(duì)于公司一個(gè)網(wǎng)絡(luò)通信服務(wù)程序使用.net core 進(jìn)行重構(gòu).重構(gòu)的目的有兩個(gè):一是讓程序能夠跨平臺(tái)運(yùn)行. 二是優(yōu)化程序代碼結(jié)構(gòu)是程序的可維護(hù)性有所提升.  重構(gòu)的過程主要由我來設(shè)計(jì)底層的架構(gòu). 在這個(gè)過程中,由于我對(duì)C# 泛型的理解還不夠深入,所以在這個(gè)方面我就犯了個(gè)錯(cuò)誤. 希望本文能把我犯的這個(gè)錯(cuò)誤闡述清楚, 如果能幫助園里其他朋友避免這個(gè)問題當(dāng)然是最好的了.

早前的設(shè)計(jì)

先用一張圖來描述早前的代碼結(jié)構(gòu)

C#泛型設(shè)計(jì)需要注意什么

Singleton<T> :是一個(gè)單例的基類, 用來實(shí)現(xiàn)單例模式.

Base<T> : 則是一個(gè)基礎(chǔ)類,它有一些靜態(tài)的屬性和方法(例如訪問Redis,kafka,數(shù)據(jù)庫等). 這些屬性和方法提供給 Child1 和 Child2 去使用.

Child1 和Child2: 相當(dāng)于不同模塊的業(yè)務(wù)邏輯實(shí)現(xiàn).

 我期望的結(jié)果是Base<T>里面的靜態(tài)成員在整個(gè)程序運(yùn)行期間只有一份.

代碼的實(shí)現(xiàn)

Singleton

public abstract class Singleton<T> where T : new()
 {
  /// <summary>
  /// 鎖定對(duì)象
  /// </summary>
  private static readonly object locker = new object();
  /// <summary>
  /// T 的實(shí)例
  /// </summary>
  static T instance = default(T);
  /// <summary>
  /// T 的實(shí)例
  /// </summary>
  public static T Instance
  {
   get
   {
    if (null == instance)
    {
     lock (locker)
     {
      if (null == instance)
      {
       instance = new T();
      }
     }
    }
    return instance;
   }
  }
 }

Base

public class Base<T> : Singleton<T> where T : new()
{
 protected static object Object { set; get; }

 static Base()
 {
  Object = new object();
 }
}

Child1 和Child2

public class Child1 : Base<Child1>
{
}

public class Child2 : Base<Child2>
{
}

我以為 Base的靜態(tài)構(gòu)造函數(shù)只會(huì)執(zhí)行一次. 可是當(dāng)我在程序里使用 Child1.Instance Child2.Instance 時(shí)發(fā)現(xiàn), Base的靜態(tài)構(gòu)造函數(shù)被執(zhí)行了2次. 那么Child1.Instance的Object和Child2.Instance的Object對(duì)象一定不是同一個(gè).

那么問題出現(xiàn)在什么地方了呢? 答案其實(shí)挺簡單的:系統(tǒng)認(rèn)為 Base<Child1> Base<Child2>并不相同. 相當(dāng)于在系統(tǒng)里定義了Base_Child1 和Base_Child2兩個(gè)類. 如果我們這么理解這個(gè)問題 ,那么Base的靜態(tài)構(gòu)造函數(shù)被執(zhí)行了2次就不難理解了.(我覺得我已經(jīng)把這個(gè)問題的成因描述清楚了,如果您沒理解,歡迎在下面評(píng)論.)

如果要達(dá)到我設(shè)計(jì)的目標(biāo)應(yīng)該怎么做呢?

修正的設(shè)計(jì)

還是先上類圖.

C#泛型設(shè)計(jì)需要注意什么

Base:

public class Base
{
 protected static object Object { set; get; }

 static Base()
 {
  Object = new object();
 }
}

Singleton:

public abstract class Singleton<T>: Base where T : new()
{
  /// <summary>
  /// 鎖定對(duì)象
  /// </summary>
  private static readonly object locker = new object();
  /// <summary>
  /// T 的實(shí)例
  /// </summary>
  static T instance = default(T);
  /// <summary>
  /// T 的實(shí)例
  /// </summary>
  public static T Instance
  {
   get
   {
    if (null == instance)
    {
     lock (locker)
     {
      if (null == instance)
      {
       instance = new T();
      }
     }
    }
    return instance;
   }
  }
}

Child1 和Child2:

public class Child1 : Singleton<Child1>
{
}

public class Child2 : Singleton<Child2>
{
}

由Singleton 來繼承Base.然后Child1 和Child2來繼承Singleton. 這樣問題就都解決了.

以上是“C#泛型設(shè)計(jì)需要注意什么”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI