C#泛型設(shè)計需要注意的一個小陷阱
前言
距離上次發(fā)表博客已經(jīng)有幾年了. 對于沒能堅持更新博客,實在是感覺到甚是慚愧.
閑言少敘, 直接切入主題.
什么是泛型
我們在編寫程序時,經(jīng)常遇到兩個模塊的功能非常相似,只是一個是處理int數(shù)據(jù),另一個是處理string數(shù)據(jù),或者其他自定義的數(shù)據(jù)類型,但我們沒有辦法,只能分別寫多個方法處理每個數(shù)據(jù)類型,因為方法的參數(shù)類型不同。有沒有一種辦法,在方法中傳入通用的數(shù)據(jù)類型,這樣不就可以合并代碼了嗎?泛型的出現(xiàn)就是專門解決這個問題的。
但泛型就簡單嗎?當然不是,繼續(xù)往下看..
背景
最近一直在對于公司一個網(wǎng)絡(luò)通信服務(wù)程序使用.net core 進行重構(gòu).重構(gòu)的目的有兩個:一是讓程序能夠跨平臺運行. 二是優(yōu)化程序代碼結(jié)構(gòu)是程序的可維護性有所提升. 重構(gòu)的過程主要由我來設(shè)計底層的架構(gòu). 在這個過程中,由于我對C# 泛型的理解還不夠深入,所以在這個方面我就犯了個錯誤. 希望本文能把我犯的這個錯誤闡述清楚, 如果能幫助園里其他朋友避免這個問題當然是最好的了.
早前的設(shè)計
先用一張圖來描述早前的代碼結(jié)構(gòu)

Singleton<T> :是一個單例的基類, 用來實現(xiàn)單例模式.
Base<T> : 則是一個基礎(chǔ)類,它有一些靜態(tài)的屬性和方法(例如訪問Redis,kafka,數(shù)據(jù)庫等). 這些屬性和方法提供給 Child1 和 Child2 去使用.
Child1 和Child2: 相當于不同模塊的業(yè)務(wù)邏輯實現(xiàn).
我期望的結(jié)果是Base<T>里面的靜態(tài)成員在整個程序運行期間只有一份.
代碼的實現(xiàn)
Singleton
public abstract class Singleton<T> where T : new()
{
/// <summary>
/// 鎖定對象
/// </summary>
private static readonly object locker = new object();
/// <summary>
/// T 的實例
/// </summary>
static T instance = default(T);
/// <summary>
/// T 的實例
/// </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ù)只會執(zhí)行一次. 可是當我在程序里使用 Child1.Instance 和 Child2.Instance 時發(fā)現(xiàn), Base的靜態(tài)構(gòu)造函數(shù)被執(zhí)行了2次. 那么Child1.Instance的Object和Child2.Instance的Object對象一定不是同一個.
那么問題出現(xiàn)在什么地方了呢? 答案其實挺簡單的:系統(tǒng)認為 Base<Child1> 和 Base<Child2>并不相同. 相當于在系統(tǒng)里定義了Base_Child1 和Base_Child2兩個類. 如果我們這么理解這個問題 ,那么Base的靜態(tài)構(gòu)造函數(shù)被執(zhí)行了2次就不難理解了.(我覺得我已經(jīng)把這個問題的成因描述清楚了,如果您沒理解,歡迎在下面評論.)
如果要達到我設(shè)計的目標應(yīng)該怎么做呢?
修正的設(shè)計
還是先上類圖.

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>
/// 鎖定對象
/// </summary>
private static readonly object locker = new object();
/// <summary>
/// T 的實例
/// </summary>
static T instance = default(T);
/// <summary>
/// T 的實例
/// </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. 這樣問題就都解決了.
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
C#實現(xiàn)判斷文件夾存在與否并創(chuàng)建文件夾的方法
這篇文章主要介紹了C#實現(xiàn)判斷文件夾存在與否并創(chuàng)建文件夾的方法,涉及C#針對文件及目錄的判斷與創(chuàng)建操作相關(guān)技巧,需要的朋友可以參考下2017-02-02
C#中將字符串轉(zhuǎn)換為整型的三種解決方法總結(jié)
本篇文章是對C#中將字符串轉(zhuǎn)換為整型的三種解決方法進行了詳細的分析介紹,需要的朋友參考下2013-06-06
C# 制作PictureBox圓形頭像框并從數(shù)據(jù)庫中讀取頭像
C#提供的PictureBox控鍵默認情況下是方形的非常大的影響美觀,怎么解決這一問題呢?下面小編給大家?guī)砹薈# 制作PictureBox圓形頭像框并從數(shù)據(jù)庫中讀取頭像的操作代碼,感興趣的朋友一起學習下吧2021-08-08

