.NET的资源并不限于.resx文件(二)
ResourceManager在默认的情况下只能提供对内嵌于程序集的.resources资源文件的存取。
为了实现对独立二进制.resources资源文件的支持,我们自定义了BinaryResoruceNManager。
在本篇中将创建两个自定义的ResourceManager,以实现对独立.resx资源文件和自定义结构的XML资源文件的支持。
一、自定义ResXResourceManager实现对.Resx资源文件的支持
较之.resources资源文件这种二进制文件,以XML形式定义的.Resx资源文件是一个纯文本文件,我们可以对其进行自由地修改,所以有时候我们直接将独立的.resx文件作为资源存储形式更利于资源内容的维护。我们创建自定义的BinaryResourceManager实现了对独立.resources资源文件的支持,我们仅仅需要采用相似的方式定义一个ResXResourceManager。由于.NET已经提供了支持.Resx资源文件的ResourceSet、ResourceReader和ResourceWriter,所以ResXResourceManager和BinaryResourceManager一样简单,下面是其全部定义。
1: public class ResXResourceManager : FileResourceManager
2: {
3: public ResXResourceManager(string directory, string baseName)
4: : base(directory, baseName, ".resx")
5: {}
6:
7: protected override ResourceSet InternalGetResourceSet(CultureInfo culture, bool createIfNotExists, bool tryParents)
8: {
9: return new ResXResourceSet(this.GetResourceFileName(culture));
10: }
11: }
.resx文件对应的ReourceSet为ResXResourceSet,定义在System.Windows.Forms程序集中,所以在重写的InternalGetResourceSet中我们只需要返回这么一个ResXResourceSet即可。ResXResourceSet对应的ResourceReader为ResXResourceReader,而.resx文件可以通过ResXResourceWriter进行写入。
既然我们的ResXResourceManager已经创建好了,我们就可以将它应用到我们的演示程序中。演示代码如下所示,三个辅助方法PrepareFiles、AddResource和DisplayResource的实现可以参考《上篇》,后面列出的是与之前的演示完全一样的输出结果。
1: PrepareFiles("GreetingMessages", "resx");
2:
3: AddResource(() => new ResXResourceWriter("GreetingMessages.resx"), new CultureInfo("en-US"));
4: AddResource(() => new ResXResourceWriter("GreetingMessages.en-US.resx"), new CultureInfo("en-US"));
5: AddResource(() => new ResXResourceWriter("GreetingMessages.zh-CN.resx"), new CultureInfo("zh-CN"));
6:
7: DisplayResource(new ResXResourceManager("", "GreetingMessages"));
输出结果
1: English (United States)
2: Merry Christmas!
3: Happy Chinese New Year!
4:
5: Chinese (Simplified, PRC)
6: 圣诞快乐!
7: 新年快乐!
8:
9: Japanese (Japan)
10: Merry Christmas!
11: Happy Chinese New Year!
二、将资源定义在自定义结构的XML文件中
.Resx资源文件本质上就是一XML文件,既然.Resx文件可以作为资源文件,我们肯定可以将资源定义在我们自定义的XML文件中。由于仅仅是作为演示,我尽可能简化这个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>
.NET的资源体系包含4个重要的对象,它们分别是ResourceManager、ResourceSet、ResourceReader和ResourceWriter。要实现将自定义结构的XML作为资源文件,我们需要自定义这四个类型。
三、为XML资源存储形式定义ResourceReader和ResourceWriter
我定义了如下一个XmlResourceReader作为读取XML资源文件的ResourceWriter。XmlResourceReader实现接口IResourceReader,在构造函数中将资源内容从XML文件中读取出来保存在一个XmlDocument对象中。在GetEnumerator方法中将该XmlDocument得内容转换成一个Hashtable,并返回该Hashtable的Enumerator。
1: public class XmlResourceReader: IResourceReader
2: {
3: public XmlDocument Document { get; private set; }
4: public XmlResourceReader(string fileName)
5: {
6: this.Document = new XmlDocument();
7: this.Document.Load(fileName);
8: }
9: public XmlResourceReader(Stream stream)
10: {
11: this.Document = new XmlDocument();
12: this.Document.Load(stream);
13: }
14: public IDictionaryEnumerator GetEnumerator()
15: {
16: Dictionary<string, string> set = new Dictionary<string, string>();
17: foreach (XmlNode item in this.Document.GetElementsByTagName("add"))
18: {
19: set.Add(item.Attributes["name"].Value, item.Attributes["value"].Value);
20: }
21: return set.GetEnumerator();
22: }
23: IEnumerator IEnumerable.GetEnumerator()
24: {
25: return GetEnumerator();
26: }
27: public void Dispose(){}
28: public void Close(){}
29: }
将资源内容写入XML文件的实现定义在如下一个名为XmlResourceWriter的文件中,它实现接口IResourceWriter。上面说过我们的XML仅仅提供对于纯文本内容的支持,在这里我们仅仅实现了value参数类型为string的AddResource方法。XmlResourceWriter的逻辑很简单,仅仅涉及到对于XmlDocument节点的添加和保存,所以在这里无需再多作介绍了。
1: public class XmlResourceWriter: IResourceWriter
2: {
3: public XmlDocument Document { get; private set; }
4: private string fileName;
5: private XmlElement root;
6:
7: public XmlResourceWriter(string fileName)
8: {
9: this.fileName = fileName;
10: this.Document = new XmlDocument();
11: this.Document.AppendChild(this.Document.CreateXmlDeclaration("1.0", "utf-8",null));
12: this.root = this.Document.CreateElement("resources");
13: this.Document.AppendChild(this.root);
14: }
15:
16: public void AddResource(string name, byte[] value)
17: {
18: throw new NotImplementedException();
19: }
20:
21: public void AddResource(string name, object value)
22: {
23: throw new NotImplementedException();
24: }
25:
26: public void AddResource(string name, string value)
27: {
28: var node = this.Document.CreateElement("add");
29: node.SetAttribute("name", name);
30: node.SetAttribute("value", value);
31: this.root.AppendChild(node);
32: }
33:
34: public void Generate()
35: {
36: using (XmlWriter writer = new XmlTextWriter(this.fileName, Encoding.UTF8))
37: {
38: this.Document.WriteTo(writer);
39: }
40: }
41: public void Dispose(){}
42: public void Close() { }
43: }
四、为XML资源存储形式定义ResourceSet
ResourceReader和ResourceWriter已经创建完毕,现在我们来创建自定义的ResourceSet:XmlResorceSet。我们定义的XmlResourceReader在构造函数中被实例化,在ReadResource方法执行过程中,它将被用于完成资源内容的读取操作,读取的结果最终用于初始化该XmlResuorceSet对象。
1: public class XmlResourceSet : ResourceSet
2: {
3: public XmlResourceSet(Stream stream)
4: {
5: this.Reader = new XmlResourceReader(stream);
6: this.Table = new Hashtable();
7: this.ReadResources();
8: }
9: public XmlResourceSet(string fileName)
10: {
11: base.Reader = new XmlResourceReader(fileName);
12: base.Table = new Hashtable();
13: this.ReadResources();
14: }
15: public override Type GetDefaultReader()
16: {
17: return typeof(XmlResourceReader);
18: }
19: public override Type GetDefaultWriter()
20: {
21: return typeof(XmlResourceWriter);
22: }
23: }
五、为XML资源存储形式定义ResourceManager
最后一部自然是创建我们自定义的ResourceManager:XmlResourceManager。和之前创建的BinaryResourceManager、ResXResourceManager一样,我们只需要重写InternalGetResourceSet方法,返回相应的ResourceSet对象即可,在这里返回的自然是上面创建的XmlResourceSet。
1: public class XmlResourceManager: FileResourceManager
2: {
3: public XmlResourceManager(string directory, string baseName)
4: : base(directory, baseName, ".xml")
5: {}
6:
7: protected override ResourceSet InternalGetResourceSet(CultureInfo culture, bool createIfNotExists, bool tryParents)
8: {
9: return new XmlResourceSet(this.GetResourceFileName(culture));
10: }
11: }
将XmlResourceManager放进我们的演示程序,你依然可以得到一样的结果
1: PrepareFiles("GreetingMessages", "xml");
2:
3: AddResource(() => new XmlResourceWriter("GreetingMessages.xml"), new CultureInfo("en-US"));
4: AddResource(() => new XmlResourceWriter("GreetingMessages.en-US.xml"), new CultureInfo("en-US"));
5: AddResource(() => new XmlResourceWriter("GreetingMessages.zh-CN.xml"), new CultureInfo("zh-CN"));
6:
7: DisplayResource(new XmlResourceManager("", "GreetingMessages"));
执行结果
1: English (United States)
2: Merry Christmas!
3: Happy Chinese New Year!
4:
5: Chinese (Simplified, PRC)
6: 圣诞快乐!
7: 新年快乐!
8:
9: Japanese (Japan)
10: Merry Christmas!
11: Happy Chinese New Year!
六、补充
XmlResourceManager的定义仅仅为你提供了一种实现自定义资源存储形式的解决方案,按照一样的思路,你可以采用其他的资源存储形式,比较有价值的应该是将资源内容定义在数据库表中。在分布式架构中,你甚至可以通过远程调用服务的方式来获取资源,不过在这种情况下,你应该考虑进行相应的缓存机制提升性能。
.NET的资源并不限于.resx文件(二)的更多相关文章
- .NET的资源并不限于.resx文件
为了构建一个轻量级的资源管理框架以满足简单的本地化(Localization)的需求,我试图直接对现有的Resource编程模型进行扩展.虽然最终没能满足我们的需求,但是这两天也算对.NET如何进行资 ...
- resx文件在X64位编译,提示“未能加载文件或程序集”的问题?
原文:resx文件在X64位编译,提示"未能加载文件或程序集"的问题? resx文件在X64位编译,提示"未能加载文件或程序集"的问题? 解答: 错误现象如下 ...
- [C++] 将 mp3 等音乐资源以资源形式嵌入 exe 文件中
引用:http://www.easyx.cn/skills/View.aspx?id=6 本文讲解怎样将 mp3 等音乐资源以资源形式嵌入 exe 文件中,并通过 mciSendString 调用.嵌 ...
- .resources文件转.resx 文件
最近在进行.net winform应用程序的反向工程,资源文件反向出来后都是.resources文件,工程编译和运行都没有问题,但.resources文件为二级制文件,无法在Visual Studio ...
- 将 mp3 等音乐资源以资源形式嵌入 exe 文件中
引用:http://www.easyx.cn/skills/View.aspx?id=6 本文讲解怎样将 mp3 等音乐资源以资源形式嵌入 exe 文件中,并通过 mciSendString 调用.嵌 ...
- C#窗体的resx文件
这些图片在项目文件中没找到,原来都存在了resx文件中. 属性界面的Image.BackgroundImage属性手动选择的图片会自动存储到resx文件中,之后这些图片源文件就可以删除了.resx中的 ...
- C#的Winform多语言实现(resx文件)
1. 简体中文 2. 繁体中文 3. 英文 下面子丰介绍一下实现的过程: 1. 为每个窗口创建相应语言的resx文件.子丰以英文为例,右键->添加->新建项->资源文件,文件名为窗口 ...
- WPF 的另类资源方式 Resources.resx
类似Winform的搞法,可以把资源放到Resources.resx中. 1.字符串 打开这个编辑器后,输入Name和Value就可以了. CS代码里面,很简单的调用: var title = W ...
- C#Winform中resx文件无效 找不到路径
问题由来 笔者因为更改了添加的图片的路径,再把路径改成图片所在的路径还是报resx文件无效,未能找到路径 问题原因 其实这个问题是因为对对象的引用修改了,但是resx文件中的应用还是没有修改.因为re ...
随机推荐
- 【转载】python3安装scrapy之windows32位爬坑
python3安装scrapy之windows32位爬坑 原创 2016年11月06日 01:38:08 标签: scrapy / windows / python / 开源框架 / 网络爬虫 早 ...
- 暑假集训第六周contest1
51Nod - 1413 权势二进制 题意:就是讲给出一个数n,让你求最少由多少个像0,1,10,11......这样的二进制数相加构成:样例n=9就是由9个二进制1相加组成,我不懂比赛的时候我为什么 ...
- 在dataframe添加1行(首行,或者尾部),且不覆盖
如果直接用下面的代码添加第1行,则会覆盖掉原来的第1行. #指定位置增加一行: df.loc[0]={'a':1,'b':2} 正确方法: 新建一个同样的 dataframe, 然后合并两个dataf ...
- 杀入红海市场 ZUK手机底气在哪?
从越来越奢华的发布会舞台屏幕,到创意越来越烧脑的邀请函,一款新手机的发布工作变得越来越系统化.何时展示.如何亮相,都成为影响一部手机情怀,甚至销售好坏的重要因素.虽然很难以一个固定标准衡量各个手 ...
- __filename意思
__filename 表示当前正在执行的脚本的文件名.它将输出文件所在位置的绝对路径,且和命令行参数所指定的文件名不一定相同. 如果在模块中,返回的值是模块文件的路径.
- FastCGI sent in stderr: "Access to the script 'XXX' has been denied (see security.limit_extensions)" 报错处理
一个新Lnmp环境在调试项目时出现如下报错: 处理: 在php.ini中设置cgi.fix_pathinfo=1(默认为0),重新加载php即可.参考链接https://blog.csdn.net/u ...
- in和exists比较
in是把外表和内表作hash 连接,而exists 是对外表作loop 循环,每次loop 循环再对内表进行查询. 一直以来认为exists 比in 效率高的说法是不准确的.如果查询的两个表大小相当, ...
- 《前端面试加分项目》系列 企业级Vue瀑布流
本文 GitHub github.com/ponkans/F2E 已收录,有一线大厂面试点思维导图,也整理了很多我的文档,欢迎Star和完善,大家面试可以参照考点复习.文末有福利~~ 前言 接水怪又来 ...
- SPI总线传输的4种模式
概述 在芯片的资料上,有两个非常特殊的寄存器配置位,分别是 CPOL (Clock POlarity)和 CPHA (Clock PHAse). CPOL配置SPI总线的极性 CPHA配置SPI总线的 ...
- Vue项目三、项目中碰到的问题详解
一.组件的划分创建 方法一: 把页面上需要复用的模块,拆分成组件.比如,页面的header.footer.面包屑.弹出框等拆分成组件.所以在src中应该有一个文件夹(components)专门放这些会 ...