如何让ASP.NET默认的资源编程方式支持非.ResX资源存储

时间:2022-04-27
本文章向大家介绍如何让ASP.NET默认的资源编程方式支持非.ResX资源存储,主要内容包括一、回顾一下之前创建的XmlResourceManager、二、创建自定义ResourceProvider、三、创建自定义ResourceProviderFactory、四、Global Resource编程、五、LocalResource编程、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。

之前写了两篇文章《.NET资源并不限于.ResX文件》(上篇、下篇),介绍了如何通过自定义ResourceManager的方式来扩展资源的存储形式。在那篇文章中,我定义了三种基于独立文件的ResourceManager(ResXResourceManager、BinaryResourceManager和XmlResoureManager)分别实现对.ResX,.Resource和.xml三种资源文件的访问。在本篇文章中我们将实现自定义ResourceManager和ASP.NET之间的集成,让ASP.NET现有的资源编程方式支持我们自定义的ResourceManager。

一、回顾一下之前创建的XmlResourceManager 二、创建自定义ResourceProvider 三、创建自定义ResourceProviderFactory 四、Global Resource编程 五、Local Resource编程

一、回顾一下之前创建的XmlResourceManager

本篇文章将会以我们之前创建的XmlResourceManager为例。通过自定义的XmlResourceManager,我们实现了将资源内容定义在了一个自定义结构的XML文件。该XML具有如下一个简单的结构。

   1: <?xml version="1.0" encoding="utf-8"?>
   2: <resources>
   3:   <add name="Greeting4Chris" value="Merry Christmas!" />
   4:   <add name="Greeting4NewYear" value="Happy Chinese New Year!" />
   5: </resources>

.ResX文件一样,为了提供多多语言的支持,我们用带有Culture Code后缀的文件名来区分资源文件所基于语言文化。如右图所示,我在一个Web Application中,定义了两套资源文件:Global Resource和Local Resource。两种资源类型的概念,不用多说你也应该知道。前者是一个全局意义的资源文件,供所有Web页共享;后者则是基于某个Web页单独使用的本地资源。

二、创建自定义ResourceProvider

要让ASP.NET现有的资源编程方式将我们添加的XML作为资源存储,必须了解ASP.NET内部采用的资源读取机制。实际上,ASP.NET在后台采用一个特殊的组件进行资源的读取,这个组件就是ResourceProvider。我们只需要创建一个基于XmlResourceManager的自定义ResourceProvider,并将它注册到ASP.NET中就可以了。

为此我创建了如下一个XmlResourceProvider,它实现了IResourceProvider。IResourceProvider具有一个GetObject方法和一个ResourceReader只读属性。前者实现了对指定键值的资源条目的读取,后者则返回一个ResourceReader。在GetObject方法中,我们利用自定义的XmlResourceManager进行资源的获取,而ResourceReader属性的实现中,我们返回的我们之前创建的XmlResourceReader。

   1: public class XmlResourceProvider:IResourceProvider
   2:    {
   3:        public XmlResourceManager ResourceManager { get; private set; }
   4:  
   5:        public XmlResourceProvider(string directory, string baseName)
   6:        {
   7:            this.ResourceManager = new XmlResourceManager(directory, baseName);
   8:        }
   9:  
  10:        public object GetObject(string resourceKey, CultureInfo culture)
  11:        {
  12:            return this.ResourceManager.GetObject(resourceKey, culture);
  13:        }
  14:  
  15:        public IResourceReader ResourceReader
  16:        {
  17:            get
  18:            { 
  19:                return new XmlResourceReader(Path.Combine(this.ResourceManager.Directory,this.ResourceManager.BaseName+".xml"));
  20:            }
  21:        }

三、创建自定义ResourceProviderFactory

XmlResourceProvider创建完毕,但是它不能直接被注册,我们需要创建一个对应的工厂类。为此,如下一个名称为XmlResourceProviderFactory类被创建出来。

   1: public class XmlResourceProviderFactory:ResourceProviderFactory
   2: {
   3:     public override IResourceProvider CreateGlobalResourceProvider(string classKey)
   4:     {
   5:         string directory = HttpContext.Current.Server.MapPath("GlobalResources");
   6:         return new XmlResourceProvider(directory, classKey);
   7:     }
   8:  
   9:     public override IResourceProvider CreateLocalResourceProvider(string virtualPath)
  10:     {
  11:         string directory = HttpContext.Current.Server.MapPath(VirtualPathUtility.GetDirectory(virtualPath));
  12:         string baseName = VirtualPathUtility.GetFileName(virtualPath);
  13:         return new XmlResourceProvider(directory, baseName);
  14:     }
  15: }

XmlResourceProviderFactory继承自抽象类ResourceProviderFactory,并实现了两个抽象方法CreateGlobalResourceProvider和CreateLocalResourceProvider。这两个方法均返回一个ResourceProvider对象,它们分别用于基于Global Resource和Local Resource的读取。在XmlResourceProviderFactory中,这两个方法均返回一个XmlResourceProvider对象。所不同的是,CreateGlobalResourceProvider方法返回的XmlResourceProvider基于的资源文件是一个存储在GlobalResources目录下指定名称(classKey)的XML文件,而CreateLocalResourceProvider返回的XmlResourceProvider基于的资源则是和当前访问.aspx文件处于同级目录下,并且名称和.aspx文件同名的XML文件。

在web.config中,XmlResourceProviderFactory通过<system.web>/<globalization>配置节进行注册,下面是相应的配置。

   1: <?xml version="1.0"?>
   2: <configuration>
   3:     <system.web>
   4:         <globalization uiCulture="zh-CN" resourceProviderFactoryType="Artech.ResourceAppBlock.XmlResourceProviderFactory, Artech.ResourceAppBlock.Lib"/>
   5:         <compilation debug="true"/></system.web>
   6: </configuration>

四、Global Resource编程

现在我们来验证以下通过ASP.NET原生的资源编程模式是否能够正常读取我们指定的XML。我们先来演示Global Resource的读取,为此我们创建了一个Web项目,并进行了如上的配置。如上面途中所示,我们在GlobalResources目录下添加了3个XML文件,其中GreetingMessages.xml作为语言文化中性的资源文件,而GreetingMessages.en-US.xml和GreetingMessages.zh-CN.xml则基于美式英语和简体中文。GreetingMessages.xml和GreetingMessages.en-US.xml具有相同的内容。

   1: <?xml version="1.0" encoding="utf-8"?>
   2: <resources>
   3:   <add name="Greeting4Chris" value="Merry Christmas!" />
   4:   <add name="Greeting4NewYear" value="Happy Chinese New Year!" />
   5: </resources>

而GreetingMessages.zh-CN.xml则为中文。

   1: <?xml version="1.0" encoding="utf-8"?>
   2: <resources>
   3:   <add name="Greeting4Chris" value="圣诞快乐!" />
   4:   <add name="Greeting4NewYear" value="新年快乐!" />
   5: </resources>

现在我们创建一个文件名称为Defualt.aspx的Web页,并在其中放置两个Label控件相应的HTML如下所示。

   1: <body>
   2:     <form id="form1" runat="server">
   3:     <div>
   4:         <asp:Label id="LabelGreeting4NewYear" runat="server"  />
   5:         <br/>
   6:         <asp:Label id="LabelGreeting4Chris" runat="server"/>
   7:     </div>
   8:     </form>
   9: </body>

在Page加载的时候,通过如下的代码将两个Label和相应的资源条目进行绑定。

   1: protected void Page_Load(object sender, EventArgs e)
   2: {
   3:     this.LabelGreeting4NewYear.Text = this.GetGlobalResourceObject("GreetingMessages", "Greeting4NewYear").ToString();
   4:     this.LabelGreeting4Chris.Text = this.GetGlobalResourceObject("GreetingMessages", "Greeting4Chris").ToString();
   5: }

在浏览器中访问页面,你会得到如下的文字。

1: 新年快乐!

2: 圣诞快乐!

如果当前的语言文化为en-US,或者其它非zh-CN,页面的文字将显示为英文。比如,你通过如下的配置将默认的UI Culture替换成en-US,你将在页面中得到如下的显示。

   1: <?xml version="1.0"?>
   2: <configuration>
   3:     <system.web>
   4:         <globalization uiCulture="en-US" resourceProviderFactoryType="Artech.ResourceAppBlock.XmlResourceProviderFactory, Artech.ResourceAppBlock.Lib"/>
   5:         <compilation debug="true"/></system.web>
   6: </configuration>

实现结果:

1: Happy Chinese New Year!

2: Merry Christmas!

实际上对于Global Resource的读取,你可以采用更为简洁的编程方式,就是以内联的方式,以<%$ Resources:ClassKey, ResourceKey%>的形式直接写在HTML中。在本例中,你可以不用编写任何代码,直接将HTML改成如下的形式即可。

   1: <body>
   2:     <form id="form1" runat="server">
   3:     <div>
   4:         <asp:Label id="LabelGreeting4NewYear" runat="server" Text="<%$ Resources:GreetingMessages, Greeting4NewYear%>" />
   5:         <br/>
   6:         <asp:Label id="LabelGreeting4Chris" runat="server" Text="<%$ Resources:GreetingMessages, Greeting4Chris%>" />
   7:     </div>
   8:     </form>
   9: </body>

五、LocalResource编程

上面演示了读取或者绑定Global Resource的编程方式,现在来看看Local Resource。Local Resource,顾名思义,就是每个.aspx页面对应一个独自使用的资源文件。按照如上图所示的结构,我为Default.aspx添加了三个本地资源文件:Default.aspx.xml、Default.aspx.zh-CN.xml和Default.aspx.en-US.xml。

由于Local Resource中的资源条没有自动和页面中某个控件的某个属性进行绑定。在本例中,我们希望资源文本自定义绑定到两个Label的Text属性上,所以我们需要在ResourceKey中加上属性名(Text)后缀,下面是基于英文和中文的内容。

Default.aspx.xml & Default.aspx.en-US.xml

   1: <?xml version="1.0" encoding="utf-8"?>
   2: <resources>
   3:   <add name="Greeting4Chris.Text" value="Merry Christmas!" />
   4:   <add name="Greeting4NewYear.Text" value="Happy Chinese New Year!" />
   5: </resources>

Default.aspx.zh-CN.xml

   1: <?xml version="1.0" encoding="utf-8"?>
   2: <resources>
   3:   <add name="Greeting4Chris.Text" value="圣诞快乐!" />
   4:   <add name="Greeting4NewYear.Text" value="新年快乐!" />
   5: </resources>

那么资源的绑定通过meta:resourcekey=”…” 直接写在定义控件的HTML中即可。所以我们的Default.aspx可以进行如下的改写,便从基于Global Resource的绑定转变成针对Local Resource的绑定。

   1: <body>
   2:     <form id="form1" runat="server">
   3:     <div>
   4:         <asp:Label id="LabelGreeting4NewYear" runat="server" meta:resourcekey="Greeting4NewYear" />
   5:         <br/>
   6:         <asp:Label id="LabelGreeting4Chris" runat="server" meta:resourcekey="Greeting4Chris" />
   7:     </div>
   8:     </form>
   9: </body>

实际上,我们也可以通过代码的方式获取本地资源,我们只需要调用Page的GetLocalResourceObject方法即可。上面针对本地资源的绑定与下面的代码是等效的。

   1: protected void Page_Load(object sender, EventArgs e)
   2: {
   3:     this.LabelGreeting4NewYear.Text = this.GetLocalResourceObject ("Greeting4NewYear.Text").ToString();
   4:     this.LabelGreeting4Chris.Text = this.GetLocalResourceObject("Greeting4Chris.Text").ToString();
   5: }

.NET的资源并不限于.resx文件,你可以采用任意存储形式[上篇] .NET的资源并不限于.resx文件,你可以采用任意存储形式[下篇] 如何在ASP.NET应用中使用自定义资源存储形式