tsx81429 发表于 2023-1-9 23:05:21

WPF 已知问题 dotnet 6 设置 InvariantGlobalization 之后将丢失默认绑定转

在设置了 InvariantGlobalization 为 true 之后,将会发现原本能正常工作的 XAML 可能就会抛出异常。本文将告诉大家此问题的原因
这是有开发者在 WPF 仓库上给我报告的 bug 我才找到的问题。问题的现象是 XAML 抛出异常,步骤有些复杂:

[*]升级到 dotnet 6 版本。 因为此问题是在 dotnet 6 下才能复现,在 dotnet 6 以下,如 dotnet 5 和 dotnet core 3.1 是没有问题的
[*]要求设置 InvariantGlobalization 为 true 的值
[*]在 XAML 绑定静态的非字符串类型的属性,例如 int 类型的属性,如以下代码
这是 MainWindow.xaml.cs 的代码:
using System.Windows;

namespace repro
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
      public MainWindow()
      {
            InitializeComponent();
      }
      public static string IWillNotCauseException { get; set; }
      public static int IWillCauseException { get; set; }
    }
}这是在 XAML 的代码
<Window x:
      xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
      xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
      xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
      xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
      xmlns:local="clr-namespace:repro"
      mc:Ignorable="d"
      Title="MainWindow" Height="450" Width="800">
    <Grid>
      <TextBlock Text="{Binding Source={x:Static local:MainWindow.IWillNotCauseException}}" />
      <TextBlock Text="{Binding Source={x:Static local:MainWindow.IWillCauseException}}" />
    </Grid>
</Window>运行之后,将会看到 XAML 抛出异常。详细请看 https://github.com/dotnet/wpf/issues/6477
抛出的异常包含以下信息
System.Globalization.CultureNotFoundException: 'Only the invariant culture is supported in globalization-invariant mode. See https://aka.ms/GlobalizationInvariantMode for more information. (Parameter 'name')原因是在 dotnet 6 设置了 InvariantGlobalization 为 true 之后,在调用 CultureInfo.GetCultureInfoByIetfLanguageTag 方法时,将抛出异常,如下图
https://img2023.cnblogs.com/blog/1080237/202301/1080237-20230109190745931-1674078475.jpg
这是符合 官方文档 描述的
如 Breaking change: Culture creation and case mapping in globalization-invariant mode 文档所述:
Starting in .NET 6 when globalization-invariant mode is enabled:

    If an app attempts to create a culture that's not the invariant culture, a CultureNotFoundException exception is thrown.因此这个问题其实是 dotnet 6 的符合预期的行为,也不是 WPF 的问题
附设置 InvariantGlobalization 为 true 的方法如下
编辑 csproj 项目文件,添加 true 到 PropertyGroup 里面,如以下代码
<PropertyGroup>
    <InvariantGlobalization>true</InvariantGlobalization>
</PropertyGroup>或者是编辑 runtimeconfig.json 文件,添加如下代码
{
    "runtimeOptions":
    {
      "configProperties":
      {
            "System.Globalization.Invariant": true
      }
    }
}参考文档:
Breaking change: Culture creation and case mapping in globalization-invariant mode
runtime/globalization-invariant-mode.md at main · dotnet/runtime · GitHub

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
页: [1]
查看完整版本: WPF 已知问题 dotnet 6 设置 InvariantGlobalization 之后将丢失默认绑定转