178

有没有办法为整个应用程序设置文化?所有当前线程和新线程?

我们将文化的名称存储在数据库中,当我们的应用程序启动时,我们执行

CultureInfo ci = new CultureInfo(theCultureString);
Thread.CurrentThread.CurrentCulture = ci;
Thread.CurrentThread.CurrentUICulture = ci;

但是,当然,当我们想在新线程中做某事时,这会“丢失”。有没有办法为整个应用程序设置CurrentCultureCurrentUICulture?所以新线程也得到了这种文化?还是每当创建一个我可以连接的新线程时都会触发一些事件?

4

10 回答 10

201

在 .NET 4.5 中,您可以使用该CultureInfo.DefaultThreadCurrentCulture属性来更改 AppDomain 的区域性。

对于 4.5 之前的版本,您必须使用反射来操纵 AppDomain 的文化。有一个私有静态字段CultureInfom_userDefaultCulture在 .NET 2.0 mscorlib 中,s_userDefaultCulture在 .NET 4.0 mscorlib 中)控制CurrentCulture如果线程没有在其自身上设置该属性时返回的内容。

这不会改变本机线程语言环境,并且以这种方式发布改变文化的代码可能不是一个好主意。不过,它可能对测试有用。

于 2011-09-24T00:25:59.320 回答
36

这被问了很多。基本上,没有,不是 .NET 4.0。ThreadPool您必须在每个新线程(或函数)开始时手动执行此操作。您也许可以将区域性名称(或仅区域性对象)存储在静态字段中,以省去访问数据库的麻烦,仅此而已。

于 2009-01-22T11:43:01.913 回答
19

如果您正在使用资源,您可以通过以下方式手动强制它:

Resource1.Culture = new System.Globalization.CultureInfo("fr"); 

在资源管理器中,有一个自动生成的代码如下:

/// <summary>
///   Overrides the current thread's CurrentUICulture property for all
///   resource lookups using this strongly typed resource class.
/// </summary>
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
internal static global::System.Globalization.CultureInfo Culture {
    get {
        return resourceCulture;
    }
    set {
        resourceCulture = value;
    }
}

现在,每次您在此资源中引用您的单个字符串时,它都会使用指定的 resourceCulture 覆盖区域性(线程或进程)。

您可以将语言指定为“fr”、“de”等,也可以将语言代码放在 en-US 的 0x0409 或 it-IT 的 0x0410 中。有关语言代码的完整列表,请参阅:语言标识符和语言环境

于 2011-11-08T01:01:37.507 回答
9

对于 .NET 4.5 及更高版本,您应该使用:

var culture = new CultureInfo("en-US");
CultureInfo.DefaultThreadCurrentCulture = culture;
CultureInfo.DefaultThreadCurrentUICulture = culture;
于 2016-12-13T13:52:18.850 回答
6

实际上你可以设置默认的线程文化和 UI 文化,但仅限于 Framework 4.5+

我放入了这个静态构造函数

static MainWindow()
{
  CultureInfo culture = CultureInfo
    .CreateSpecificCulture(CultureInfo.CurrentCulture.Name);
  var dtf = culture.DateTimeFormat;
  dtf.ShortTimePattern = (string)Microsoft.Win32.Registry.GetValue(
    "HKEY_CURRENT_USER\\Control Panel\\International", "sShortTime", "hh:mm tt");
  CultureInfo.DefaultThreadCurrentUICulture = culture;
}

并在 ValueConverter 的 Convert 方法中放置一个断点,以查看到达另一端的内容。CultureInfo.CurrentUICulture 不再是 en-US,而是变成了 en-AU,我的小技巧使它尊重 ShortTimePattern 的区域设置。

万岁,世界上一切都好!或不。传递给 Convert 方法的区域性参数仍然是en-US。呃,WTF?!但这是一个开始。至少这样

  • 您可以在应用加载时修复一次 UI 文化
  • 它总是可以从CultureInfo.CurrentUICulture
  • string.Format("{0}", DateTime.Now)将使用您自定义的区域设置

如果您不能使用 4.5 版框架,那么放弃将 CurrentUICulture 设置为 CultureInfo 的静态属性,并将其设置为您自己的类之一的静态属性。这不会修复 string.Format 的默认行为或使 StringFormat 在绑定中正常工作,然后遍历应用程序的逻辑树以重新创建应用程序中的所有绑定并设置其转换器文化。

于 2013-01-17T05:19:42.603 回答
3

DefaultThreadCurrentCulture并且DefaultThreadCurrentUICulture也存在于 Framework 4.0 中,但它们是私有的。使用反射,您可以轻松设置它们。这将影响所有CurrentCulture未明确设置的线程(也正在运行线程)。

Public Sub SetDefaultThreadCurrentCulture(paCulture As CultureInfo)
    Thread.CurrentThread.CurrentCulture.GetType().GetProperty("DefaultThreadCurrentCulture").SetValue(Thread.CurrentThread.CurrentCulture, paCulture, Nothing)
    Thread.CurrentThread.CurrentCulture.GetType().GetProperty("DefaultThreadCurrentUICulture").SetValue(Thread.CurrentThread.CurrentCulture, paCulture, Nothing)
End Sub
于 2015-01-20T09:44:50.787 回答
3

对于 ASP.NET5,即 ASPNETCORE,您可以在 中执行以下操作configure

app.UseRequestLocalization(new RequestLocalizationOptions
{
    DefaultRequestCulture = new RequestCulture(new CultureInfo("en-gb")),
    SupportedCultures = new List<CultureInfo>
    {
        new CultureInfo("en-gb")
    },
            SupportedUICultures = new List<CultureInfo>
    {
        new CultureInfo("en-gb")
    }
});

以下是提供更多信息的一系列博客文章:

于 2016-03-07T11:28:43.210 回答
3

这个答案是@rastating 的好答案的一些扩展。您可以将以下代码用于所有版本的 .NET,而无需担心:

    public static void SetDefaultCulture(CultureInfo culture)
    {
        Type type = typeof (CultureInfo);
        try
        {
            // Class "ReflectionContext" exists from .NET 4.5 onwards.
            if (Type.GetType("System.Reflection.ReflectionContext", false) != null)
            {
                type.GetProperty("DefaultThreadCurrentCulture")
                    .SetValue(System.Threading.Thread.CurrentThread.CurrentCulture,
                        culture, null);

                type.GetProperty("DefaultThreadCurrentUICulture")
                    .SetValue(System.Threading.Thread.CurrentThread.CurrentCulture,
                        culture, null);
            }
            else //.NET 4 and lower
            {
                type.InvokeMember("s_userDefaultCulture",
                    BindingFlags.SetField | BindingFlags.NonPublic | BindingFlags.Static,
                    null,
                    culture,
                    new object[] {culture});

                type.InvokeMember("s_userDefaultUICulture",
                    BindingFlags.SetField | BindingFlags.NonPublic | BindingFlags.Static,
                    null,
                    culture,
                    new object[] {culture});

                type.InvokeMember("m_userDefaultCulture",
                    BindingFlags.SetField | BindingFlags.NonPublic | BindingFlags.Static,
                    null,
                    culture,
                    new object[] {culture});

                type.InvokeMember("m_userDefaultUICulture",
                    BindingFlags.SetField | BindingFlags.NonPublic | BindingFlags.Static,
                    null,
                    culture,
                    new object[] {culture});
            }
        }
        catch
        {
            // ignored
        }
    }
}
于 2017-08-16T05:39:45.307 回答
2

这是 c# MVC 的解决方案:

  1. 首先:创建一个自定义属性和覆盖方法,如下所示:

    public class CultureAttribute : ActionFilterAttribute
    {
        public override void OnActionExecuting(ActionExecutingContext filterContext)
        {
            // Retreive culture from GET
            string currentCulture = filterContext.HttpContext.Request.QueryString["culture"];
    
            // Also, you can retreive culture from Cookie like this :
            //string currentCulture = filterContext.HttpContext.Request.Cookies["cookie"].Value;
    
            // Set culture
            Thread.CurrentThread.CurrentCulture = new CultureInfo(currentCulture);
            Thread.CurrentThread.CurrentUICulture = CultureInfo.CreateSpecificCulture(currentCulture);
        }
    }
    
  2. 第二:在App_Start中,找到FilterConfig.cs,添加这个属性。(这适用于整个应用程序)

    public class FilterConfig
    {
        public static void RegisterGlobalFilters(GlobalFilterCollection filters)
        {
            // Add custom attribute here
            filters.Add(new CultureAttribute());
        }
    }    
    

就是这样 !

如果你想为每个控制器/动作而不是整个应用程序定义文化,你可以像这样使用这个属性:

[Culture]
public class StudentsController : Controller
{
}

或者:

[Culture]
public ActionResult Index()
{
    return View();
}
于 2016-11-22T15:44:08.790 回答
2

为所有线程和窗口设置 CultureInfo 的工作解决方案。

  1. 打开App.xaml文件并添加一个新的“启动”属性来为应用分配启动事件处理程序:
<Application ........
             Startup="Application_Startup"
>
  1. 打开App.xaml.cs文件并将此代码添加到创建的启动处理程序(在本例中为 Application_Startup)。类 App 将如下所示:
    public partial class App : Application
    {
        private void Application_Startup(object sender, StartupEventArgs e)
        {
            CultureInfo cultureInfo = CultureInfo.GetCultureInfo("en-US");
            System.Globalization.CultureInfo.DefaultThreadCurrentCulture = cultureInfo;
            System.Globalization.CultureInfo.DefaultThreadCurrentUICulture = cultureInfo;
            Thread.CurrentThread.CurrentCulture = cultureInfo;
            Thread.CurrentThread.CurrentUICulture = cultureInfo;
        }
    }
于 2020-04-21T11:11:47.187 回答