更改.NET中的默認時區(qū)的操作方法
如何更改.NET中的默認時區(qū)?
除了"在操作系統(tǒng)中修改時區(qū)信息,然后重啟.NET應用程序,使其生效"之外。如何在不修改操作系統(tǒng)時區(qū)的前提下,修改.NET中的默認時區(qū)呢?
這是一位 同學兼同事 于5月21日在技術群里問的問題,我當時簡單地研究了一下,就寫出來了。
現在寫文章分享給大家,雖然我覺得這種需求非常小眾,幾乎不會有人用到。
正文
正常手段下,.NET是不允許開發(fā)者修改默認時區(qū)的,它沒有公開這樣的API。
在 .NET 中,管理時區(qū)的類型叫 TimeZoneInfo
,它位于 System
命名空間下,由 System.Private.CoreLib.dll
提供。
使用 ILSpy
反編譯 System.Private.CoreLib.dll
,找到 TimeZoneInfo
類型,我們可以看到 TimeZoneInfo.Local
指向一個私有字段 s_cachedData
的成員屬性 Local
,該字段類型是一個屬于 TimeZoneInfo
的私有嵌套類型 CachedData
。
當首次訪問 CachedData.Local
時,它會先檢查 _localTimeZone
私有字段是否有值。如果沒有值,則調用 CreateLocal
方法從操作系統(tǒng)獲取時區(qū)信息并且賦值。
看到了這里,我腦海里就浮現了兩種方案:
- 使用
hook
技術挾持并修改win32 api
返回的時區(qū)信息。 - 使用
reflection
技術反射并且修改時區(qū)信息。
方案1的優(yōu)點是穩(wěn)定,但可能會被殺毒軟件報毒。
方案2的優(yōu)點是不會報毒,但可能不穩(wěn)定。
為什么說方案2不穩(wěn)定呢?因為 s_cachedData 私有字段值有可能在某個時候被重置。
現在我們來看看方案2的實現:
public static bool TrySetLocalTimeZoneInfo(TimeZoneInfo timeZoneInfo) { Type timeZoneInfoType = typeof(TimeZoneInfo); // 獲取TimeZoneInfo類型的私有靜態(tài)字段成員信息s_cachedData FieldInfo cachedDataFieldInfo = timeZoneInfoType.GetField("s_cachedData", BindingFlags.NonPublic | BindingFlags.Static); if (cachedDataFieldInfo == null) { return false; } // 獲取TimeZoneInfo類型的私有嵌套類型CachedData Type cachedDataType = timeZoneInfoType.GetNestedType("CachedData", BindingFlags.NonPublic); if (cachedDataType == null) { return false; } // 獲取CachedData類型的私有字段成員信息_localTimeZone FieldInfo localTimeZoneFieldInfo = cachedDataType.GetField("_localTimeZone", BindingFlags.NonPublic | BindingFlags.Instance); if (localTimeZoneFieldInfo == null) { return false; } // 獲取TimeZoneInfo類型的私有靜態(tài)字段s_cachedData值 object cachedData = cachedDataFieldInfo.GetValue(null); if (cachedData == null) { return false; } // 設置私有字段的值 localTimeZoneFieldInfo.SetValue(cachedData, timeZoneInfo); return true; }
PS: 該方法代碼實際測試在 .NET Core 3.1
, .NET 5.0
, .NET 6.0
, .NET 7.0
, .NET 8.0
都可以正常工作。
用法:
void Main() { // 設置前 Console.WriteLine(TimeZoneInfo.Local); // 修改為 GMT 時區(qū) TimeZoneInfo hkTimeZoneInfo = TimeZoneInfo.FindSystemTimeZoneById("GMT Standard Time"); bool setResult = TrySetLocalTimeZoneInfo(hkTimeZoneInfo); // 設置后 Console.WriteLine(TimeZoneInfo.Local); }
注意:這種方案需要嚴謹測試,反復驗證。
因為是篡改.NET內部私有變量,不知道是否會引起其它后果。
比如.NET內部其它API沒有使用 TimeZoneInfo.Local
,而是自己在其它地方又緩存了一套 TimeZoneInfo
,那就GG了。
又比如,需要檢查整個 .NET Runtime
和其它第三方組件,是否有調用 TimeZoneInfo.ClearCachedData
靜態(tài)方法 或者 調用 CultureInfo.ClearCachedData
對象方法。
到此這篇關于如何更改.NET中的默認時區(qū)?的文章就介紹到這了,更多相關.NET默認時區(qū)內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
比較簡單的將數據信息導入wrod文檔方案(C# for word)
史上最簡單將數據信息導入wrod文檔方案(C# for word)2010-01-01DataSet、DataTable、DataRow區(qū)別詳解
這篇文章主要介紹了DataSet、DataTable、DataRow區(qū)別詳解,需要的朋友可以參考下2014-02-02