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

之前写了两篇文章《.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应用中使用自定义资源存储形式

作者:蒋金楠
微信公众账号:大内老A
微博:www.weibo.com/artech
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号(原来公众帐号蒋金楠的自媒体将会停用)。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

原文链接

时间: 2024-09-13 05:18:35

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

一起谈.NET技术,如何让ASP.NET默认的资源编程方式支持非.ResX资源存储

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

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

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

1线程概念:线程和进程之间的关系,线程间可共享资源,线程间非共享资源,线程的优缺点

 1线程概念 1.1什么是线程 1.1.2线程和进程的关系         1.轻量级进程(light-weightprocess),也有PCB,创建线程使用底层函数和进程一样,都是clone. 2.从内核里看进程和线程是一样的,都有各自不同的PCB,但是PCB中指向内存资源的三级页表示相同的 3.进程可以蜕变成线程 4.在美国人眼里,线程就是寄存器和栈 5.在Linux下,线程是最小的执行单位:进程是最小的分配资源的单位   查看LWP号(通过这个命令可以查看到线程号),命令是: ps –

ASP.NET MVC的四种验证编程方式_实用技巧

我们可以采用4种不同的编程模式来进行针对绑定参数的验证. 一.手工验证绑定的参数 在定义具体Action方法的时候,对已经成功绑定的参数实施手工验证无疑是一种最为直接的编程方式,接下来我们通过一个简单的实例来演示如何将参数验证逻辑实现在对应的Action方法中,并在没有通过验证的情况下将错误信息响应给客户端.我们在一个ASP.NET MVC应用中定义了如下一个Person类作为被验证的数据类型,它的Name.Gender和Age三个属性分别表示一个人的姓名.性别和年龄. public class

ASP.NET MVC下的四种验证编程方式[续篇]_实用技巧

在<ASP.NET MVC的四种验证编程方式>一文中我们介绍了ASP.NET MVC支持的四种服务端验证的编程方式("手工验证"."标注ValidationAttribute特性"."让数据类型实现IValidatableObject或者IDataErrorInfo"),那么在ASP.NET MVC框架内部是如何提供针对这四种不同编程方式的支持的呢?接下来我们就来聊聊这背后的故事. 一.ModelValidator与ModelVali

ASP.NET MVC下的四种验证编程方式

原文:ASP.NET MVC下的四种验证编程方式 ASP.NET MVC采用Model绑定为目标Action生成了相应的参数列表,但是在真正执行目标Action方法之前,还需要对绑定的参数实施验证以确保其有效性,我们将针对参数的验证成为Model绑定.总地来说,我们可以采用4种不同的编程模式来进行针对绑定参数的验证. 目录 一.手工验证绑定的参数 二.使用ValidationAttribute特性 三.让数据类型实现IValidatableObject接口 四.让数据类型实现IDataError

ASP.NET MVC下的四种验证编程方式[续篇]

原文:ASP.NET MVC下的四种验证编程方式[续篇] 在<ASP.NET MVC下的四种验证编程方式>一文中我们介绍了ASP.NET MVC支持的四种服务端验证的编程方式("手工验证"."标注ValidationAttribute特性"."让数据类型实现IValidatableObject或者IDataErrorInfo"),那么在ASP.NET MVC框架内部是如何提供针对这四种不同编程方式的支持的呢?接下来我们就来聊聊这背后的

ASP.NET程序中常用编程代码

ASP.NET程序中常用编程代码 1.为按钮添加确认对话框 Button1.Attributes.Add("onclick","return confirm('确认?')");button.attributes.add("onclick","if(confirm('are you sure...?')){return true;}else{return false;}") 2.删除表格选定记录 //获得DataGrid主键in

如何在不影响asp.net默认安全性的前提下使用ckeditor/fckeditor?

asp.net默认情况下,不允许提交包含html源代码的表单,这在很大程度上防止了跨站(提交)攻击,但是ckeditor/fckeditor之类的富文本编辑器肯定是要生成html源代码的,如何解决这个矛盾? 通常的办法是修改web.config asp.net2.0/3/3.5时可以这样做: <pages validateRequest="false"></pages> asp.net4.0下,这样还不够,必须写成这样: <pages validateRe