经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » ASP.net » 查看文章
如何更改.NET中的默认时区?
来源:cnblogs  作者:VAllen  时间:2024/6/17 15:15:15  对本文有异议

除了"在操作系统中修改时区信息,然后重启.NET应用程序,使其生效"之外。如何在不修改操作系统时区的前提下,修改.NET中的默认时区呢?

这是一位 同学兼同事 于5月21日在技术群里问的问题,我当时简单地研究了一下,就写出来了。

现在写文章分享给大家,虽然我觉得这种需求非常小众,几乎不会有人用到。

正文

正常手段下,.NET是不允许开发者修改默认时区的,它没有公开这样的API。

在 .NET 中,管理时区的类型叫 TimeZoneInfo,它位于 System 命名空间下,由 System.Private.CoreLib.dll 提供。

使用 ILSpy 反编译 System.Private.CoreLib.dll,找到 TimeZoneInfo 类型,我们可以看到 TimeZoneInfo.Local 指向一个私有字段 s_cachedData 的成员属性 Local,该字段类型是一个属于 TimeZoneInfo 的私有嵌套类型 CachedData

当首次访问 CachedData.Local 时,它会先检查 _localTimeZone 私有字段是否有值。如果没有值,则调用 CreateLocal 方法从操作系统获取时区信息并且赋值。

看到了这里,我脑海里就浮现了两种方案:

  1. 使用 hook 技术挟持并修改 win32 api 返回的时区信息。
  2. 使用 reflection 技术反射并且修改时区信息。

方案1的优点是稳定,但可能会被杀毒软件报毒。

方案2的优点是不会报毒,但可能不稳定。

为什么说方案2不稳定呢?因为 s_cachedData 私有字段值有可能在某个时候被重置。

现在我们来看看方案2的实现:

  1. public static bool TrySetLocalTimeZoneInfo(TimeZoneInfo timeZoneInfo)
  2. {
  3. Type timeZoneInfoType = typeof(TimeZoneInfo);
  4. // 获取TimeZoneInfo类型的私有静态字段成员信息s_cachedData
  5. FieldInfo cachedDataFieldInfo = timeZoneInfoType.GetField("s_cachedData", BindingFlags.NonPublic | BindingFlags.Static);
  6. if (cachedDataFieldInfo == null)
  7. {
  8. return false;
  9. }
  10. // 获取TimeZoneInfo类型的私有嵌套类型CachedData
  11. Type cachedDataType = timeZoneInfoType.GetNestedType("CachedData", BindingFlags.NonPublic);
  12. if (cachedDataType == null)
  13. {
  14. return false;
  15. }
  16. // 获取CachedData类型的私有字段成员信息_localTimeZone
  17. FieldInfo localTimeZoneFieldInfo = cachedDataType.GetField("_localTimeZone", BindingFlags.NonPublic | BindingFlags.Instance);
  18. if (localTimeZoneFieldInfo == null)
  19. {
  20. return false;
  21. }
  22. // 获取TimeZoneInfo类型的私有静态字段s_cachedData值
  23. object cachedData = cachedDataFieldInfo.GetValue(null);
  24. if (cachedData == null)
  25. {
  26. return false;
  27. }
  28. // 设置私有字段的值
  29. localTimeZoneFieldInfo.SetValue(cachedData, timeZoneInfo);
  30. return true;
  31. }

PS: 该方法代码实际测试在 .NET Core 3.1, .NET 5.0, .NET 6.0, .NET 7.0, .NET 8.0 都可以正常工作。

用法:

  1. void Main()
  2. {
  3. // 设置前
  4. Console.WriteLine(TimeZoneInfo.Local);
  5. // 修改为 GMT 时区
  6. TimeZoneInfo hkTimeZoneInfo = TimeZoneInfo.FindSystemTimeZoneById("GMT Standard Time");
  7. bool setResult = TrySetLocalTimeZoneInfo(hkTimeZoneInfo);
  8. // 设置后
  9. Console.WriteLine(TimeZoneInfo.Local);
  10. }

注意:这种方案需要严谨测试,反复验证。

因为是篡改.NET内部私有变量,不知道是否会引起其它后果。

比如.NET内部其它API没有使用 TimeZoneInfo.Local,而是自己在其它地方又缓存了一套 TimeZoneInfo,那就GG了。

又比如,需要检查整个 .NET Runtime 和其它第三方组件,是否有调用 TimeZoneInfo.ClearCachedData 静态方法 或者 调用 CultureInfo.ClearCachedData 对象方法。

原文链接:https://www.cnblogs.com/VAllen/p/18251617/How-to-change-Default-Time-Zone-in-dotnet

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号