扫一扫
分享文章到微信
扫一扫
关注官方公众号
至顶头条
在我的网站中,在创建资源文件并加入一些本地化数据后,我首先开始使用显式本地化来设置控件(例如,在我的网站中的标签)的文本,以便它们可以从资源文件中得到它们的值。既然存在四种语言;所以,除一个完全可依赖的资源文件之外(没有本地化命名),我创建了四个资源文件。
注意,这些资源文件都以本地化标记作为它们的中间名称,因此,我需要把UICulture设置为与该本地化相同的名字以便ASP.NET存取这些资源文件。
但是,问题是:我该怎样在PostBack事件中动态地改变文化呢?幸好,ASP.NET在Page类中提供了一种可重载的方法: InitializeCulture()。这个方法在页面生命周期(在生成任何控件之前)中执行得很早,并且在此,我们能够设置当前线程的UICulture和Culture。
由于这个方法位于Page类中,并且我不想针对每一个web页面都重复相同的代码,所以我创建了一个BasePage类,我的应用程序中的所有的aspx页面都派生自这个BasePage类。但是现在,我又面临另一个问题。下面,让我进行解释:
回到UI设计:我使用了一个MasterPage和一个Header用户控件(在一个ContentPlaceHolder内)。我把一个缺省的页面与该MasterPage相关联。整个站点必须动态地实现本地化。因此,在顶部,有一个下拉框,用户可以从中选择一种语言/文化。在BasePage的InitilializeCulture方法中,我必须取得用户从下拉框选择的项的值;但是,因为它还没有被初始化,所以,我还不能存取任何控件的值。答案是:使用表单集合(从响应对象内)。下面是实现代码:
///<SUMMARY> ///从通用的页面头部的下拉框列表中选择的语言名。 ///我们需要使用这个名字,因为我们还没有任何其它控件属性-现在控件本身还没有被初始化。 ///因此,我们使用"嵌套的"下拉框列表名,从中我们可以从Request.Form[]集合中得到该下拉框列表的值。 /// </SUMMARY> public const string LanguageDropDownID = "ctl00$cphHeader$Header1$ddlLanguage"; /// <SUMMARY> ///在一个回寄表单中的PostBack事件目标域的名字。你可以使用 ///它来确定是哪个控件触发了PostBack: /// Request.Form[PostBackEventTarget] . /// </SUMMARY> public const string PostBackEventTarget = "__EVENTTARGET"; |
/// <SUMMARY> ///重载InitializeCulture方法来设置在当前线程中用户选择的选项 ///。注意,这个方法在Page生命周期的早期调用 ///,并且目前我们不存在任何控件 ///,因此必须使用Form集合. /// </SUMMARY> protected override void InitializeCulture() { ///<remarks><REMARKS> ///检查是否PostBack发生.不能使用在此方法中使用IsPostBack ///,因为这个属性还没有设置。 ///</remarks> if (Request[PostBackEventTarget] != null) { string controlID = Request[PostBackEventTarget]; if (controlID.Equals(LanguageDropDownID)) { string selectedValue = Request.Form[Request[PostBackEventTarget]].ToString(); switch (selectedValue) { case "0": SetCulture("hi-IN", "hi-IN"); break; case "1": SetCulture("en-US", "en-US"); break; case "2": SetCulture("en-GB", "en-GB"); break; case "3": SetCulture("fr-FR", "fr-FR"); break; default: break; } } } ///<remarks> ///从会话中取得文件,如果控制给导航到同一程序中的一个新页面。 ///</remarks> if (Session["MyUICulture"] != null && Session["MyCulture"] != null) { Thread.CurrentThread.CurrentUICulture = (CultureInfo)Session["MyUICulture"]; Thread.CurrentThread.CurrentCulture = (CultureInfo)Session["MyCulture"]; } base.InitializeCulture(); } /// <Summary> ///使用参数设置当前的UICulture和CurrentCulture /// </Summary> /// <PARAM name="name"></PARAM> /// <PARAM name="locale"></PARAM> protected void SetCulture(string name, string locale) { Thread.CurrentThread.CurrentUICulture = new CultureInfo(name); Thread.CurrentThread.CurrentCulture = new CultureInfo(locale); ///<remarks> ///由用户把当前线程的文化集保存在会话中 ///,以便它能够在当前应用程序中跨页面应用。 ///</remarks> Session["MyUICulture"] = Thread.CurrentThread.CurrentUICulture; Session["MyCulture"] = Thread.CurrentThread.CurrentCulture; } |
因此,用户在他/她选择的语言中会看到此内容。我们需要把该文件选择保存到一个会话或一个Cookie变量中,因为如果用户移动到同一应用程序中的其它一些页面,那么,当新的页面类一开始被实例化时,该线程的文化信息将会丢失(HTTP是无状态的!)。注意,在用户的会话到期时,如果你不想丢失当前线程的文化信息,那么你可以使用Cookies。请注意,在此,我是如何使用"parentControl:ChildControl"方法从表单集合中存取控件的。通过使用这一约定,你可以存取任何ASP.NET生成的嵌套控件。借助于表单集合中选择的值,我可以通过一个switch case语句来进行文化设置:
/// <SUMMARY> ///重载InitializeCulture方法来设置在当前线程中用户选择的选项 ///。注意,这个方法在Page生命周期的早期调用 ///,并且目前我们不存在任何控件 ///,因此必须使用Form集合. /// </SUMMARY> protected override void InitializeCulture() { ///<remarks><REMARKS> ///检查是否PostBack发生.不能使用在此方法中使用IsPostBack ///,因为这个属性还没有设置。 ///</remarks> if (Request[PostBackEventTarget] != null) { string controlID = Request[PostBackEventTarget]; if (controlID.Equals(LanguageDropDownID)) { string selectedValue = Request.Form[Request[PostBackEventTarget]].ToString(); switch (selectedValue) { case "0": SetCulture("hi-IN", "hi-IN"); break; case "1": SetCulture("en-US", "en-US"); break; case "2": SetCulture("en-GB", "en-GB"); break; case "3": SetCulture("fr-FR", "fr-FR"); break; default: break; } } } ///<remarks> ///从会话中取得文件,如果控制给导航到同一程序中的一个新页面。 ///</remarks> if (Session["MyUICulture"] != null && Session["MyCulture"] != null) { Thread.CurrentThread.CurrentUICulture = (CultureInfo)Session["MyUICulture"]; Thread.CurrentThread.CurrentCulture = (CultureInfo)Session["MyCulture"]; } base.InitializeCulture(); } /// <Summary> ///使用参数设置当前的UICulture和CurrentCulture /// </Summary> /// <PARAM name="name"></PARAM> /// <PARAM name="locale"></PARAM> protected void SetCulture(string name, string locale) { Thread.CurrentThread.CurrentUICulture = new CultureInfo(name); Thread.CurrentThread.CurrentCulture = new CultureInfo(locale); ///<remarks> ///由用户把当前线程的文化集保存在会话中 ///,以便它能够在当前应用程序中跨页面应用。 ///</remarks> Session["MyUICulture"] = Thread.CurrentThread.CurrentUICulture; Session["MyCulture"] = Thread.CurrentThread.CurrentCulture; } |
因此,用户在他/她选择的语言中会看到此内容。我们需要把该文件选择保存到一个会话或一个Cookie变量中,因为如果用户移动到同一应用程序中的其它一些页面,那么,当新的页面类一开始被实例化时,该线程的文化信息将会丢失(HTTP是无状态的!)。注意,在用户的会话到期时,如果你不想丢失当前线程的文化信息,那么你可以使用Cookies。 在此,我们添加了一个新域:CultureID,它等价于LCID(或Locale标识符)。我们能够按如下所示添加文化特定的本地化数据:
现在,我们可以使用以CultureID(LCID)作为参数的SQL查询来取得本地化内容。我们还能够提供一个用户接口来把本地化数据输入到这样的表格以便能够以一种交互方式创建相应的内容。
四、总结
在本文中,我们讨论了在ASP.NET 2.0开发中有关实现全球化的一些重要方面,并且看到,这是非常容易实现的事情;但是,也存在许多值得注意的重要问题:
1.不要依赖于web浏览器的设置。你可以在应用程序中显示一个链接(可以在头部位置)以便用户能够通过点击它来选择他们的语言。
2.使用资源文件来把GUI中与描述相关的数据分离开来。资源fallback是ASP.NET使用的方法-当它不能找到相应于一种特定文化的资源文件时。它将首先试用中立资源文件,然后是缺省的资源文件或fallback资源文件(TestSiteResource.resx)。
3.使用数据库表格把数据存储到一个DB中。为此,你需要创建单独的表格来存储本地化内容。
4.如果你使用sn.exe来为你的主应用程序程序集创建一个强类型名,那么,你需要使用同一对密钥对(由sn.exe生成)中的私钥来签名你的小程序集;因为,强类型名字的程序集要求小程序集也应该是强类型名字。
如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。
现场直击|2021世界人工智能大会
直击5G创新地带,就在2021MWC上海
5G已至 转型当时——服务提供商如何把握转型的绝佳时机
寻找自己的Flag
华为开发者大会2020(Cloud)- 科技行者