目       录

第十三章     中英文版本切换设计... 2

13.1        不用自带的资源文件的理由... 2

13.2        配置文件... 2

13.3        语言管理类... 3

13.4        应用管理类... 12

13.5        小结... 12

第十三章      中英文版本切换设计

13.1     不用自带的资源文件的理由

可以利用resx资源文件进行多语言设计,resx文件本身是kv类型的资源文件,设计好资源文件后,启动软件时可以通过CurrentCulture属性设置要显示的语言。实现代码如下:

//设置成英文版本

Thread.CurrentThread.CurrentCulture = CultureInfo.GetCultureInfo("en-us");

但是,软件涉及到多线程、线程池、异步等应用的时候,当前线程设置了英文版本,其他线程还是默认的语言文化,例如:主线程设置了en-US,但是新建线程和其他已经存在的线程还是zh-CN,如果各部分UI不在同一线程更新的话,语言文化的设置是不一样的,所以没有办法实现统一的语言显示。

那么,可不可以通过进程获得所有线程信息,统一进行设置语言文化信息呢,的确是一个很好的想法。但是,通过实践证明这是行不能的,可能造反软件异常退出。为什么会出现这个现象呢?我猜想,一个进程中不仅包括自定义的线程,还存在系统级的线程,这样操作是一件危险的事。

难道就没有办法实现了吗?人不可能被尿憋死。在.NET 4.5中就简单多了,直接使用System.Globalization命名空间内CultureInfo类型的 DefaultThreadCurrentCulture和DefaultThreadCurrentUICulture属性。设置好后,每一个新线程的 CurrentUICulture和CurrentCulture属性都会和这个值保持一致的。CultureInfo类具体怎么实现的,还没有研究过。

为了兼容XP操作系统,还在使用.NET4.0的框架。相信也可以实现CultureInfo类的功能,但是不如自己设计一套语言版本方案更直接、更省时间。有时间的情况下可以研究一下CultureInfo类的实现。

13.2     配置文件

先设计语言配置文件,文件格式采用XML,存储方式采用KV的方式,文件命名可以自定义,例如:cn.xml、en.xml。如下图:

Key的定义有两种方式,第一种:窗体命名.控件命名,可以对窗体的控件统一改变显示的语言信息。第二种:直接定义关键字,可以对提示信息、状态信息等单独词条改变显示的语言信息。Value就是最终要显示语言的具体内容,完全自定义。

13.3     语言管理类

  1. 定义一个词条对应的可序列化的类。代码如下:
[Serializable]
public class CultureItem
{
/// <summary>
/// 控件的级联ID,中间用"."分隔
/// </summary>
[XmlAttribute]
public string Key { set; get; } /// <summary>
/// 中文或英文描述
/// </summary>
[XmlAttribute]
public string Value { set; get; }
}
  1. 定义一个设置语言属性的枚举。代码如下:
public enum CultureLanguage
{
[EnumDescription("中文")]
Chinese,
[EnumDescription("英文")]
English
}
  1. 开发一个语言管理类库,本质上是根据语言配置文件对Dictionary<string, string>字典缓存进行操作。实现代码如下:
public class CultureMananger
{
private static Dictionary<string, string> _dic = new Dictionary<string, string>();
private static string _cnPath = Application.StartupPath + "\\SuperIO\\Language\\cn.xml";
private static string _enPath = Application.StartupPath + "\\SuperIO\\Language\\en.xml";
private static object SyncObject = new object(); /// <summary>
/// 加载语言文件到缓存中
/// </summary>
public static void LoadCulture()
{
lock (SyncObject)
{
if (IsLanguage)
{
try
{
_dic.Clear();
string path = String.Empty;
if (Language == CultureLanguage.Chinese)
{
path = _cnPath;
}
else if (Language == CultureLanguage.English)
{
path = _enPath;
}
if (File.Exists(path))
{
List<CultureItem> itemList =SerializeOperation.SerializeOperation.GetSerialize<List<CultureItem>>(path);
foreach (CultureItem item in itemList)
{
_dic.Add(item.Key, item.Value);
}
}
}
catch (Exception ex)
{
GeneralLog.WriteLog(ex);
}
}
}
} /// <summary>
/// 清除缓存中的语言信息
/// </summary>
public static void ClearCache()
{
lock (SyncObject)
{
_dic.Clear();
}
} /// <summary>
/// 设置和获得语言类型属性
/// </summary>
public static CultureLanguage Language
{
set
{
if (GlobalProperty.GetInstance().Language != value)
{
GlobalProperty.GetInstance().Language = value;
GlobalProperty.GetInstance().Save();
LoadCulture();
}
}
get { return GlobalProperty.GetInstance().Language; }
} /// <summary>
/// 获得词条对应的描述信息
/// </summary>
/// <param name="formName">窗体名称</param>
/// <param name="field">词条字段</param>
/// <returns>对应的描述信息</returns>
public static string GetString(string formName, string field)
{
return GetString(String.Format("{0}.{1}", formName, field));
} /// <summary>
/// 获得词条对应的描述信息
/// </summary>
/// <param name="key">字段的关键字</param>
/// <returns></returns>
public static string GetString(string key)
{
lock (SyncObject)
{
if (IsLanguage)
{
string val = String.Empty;
if (_dic.ContainsKey(key))
{
_dic.TryGetValue(key, out val);
}
return val;
}
else
{
return String.Empty;
}
}
} /// <summary>
/// 应用窗体,改变语言显示
/// </summary>
/// <param name="frm"></param>
public static void ApplyResourcesForm(Form frm)
{
if (IsLanguage)
{
string frmText = GetString(frm.Name);
if (!String.IsNullOrEmpty(frmText))
{
frm.Text = frmText;
}
ApplyControls(frm.Name, frm.Controls);
}
} /// <summary>
/// 应用BarManager工具具,改变语言显示
/// </summary>
/// <param name="name"></param>
/// <param name="bar"></param>
public static void AppResourceBarItem(string name, BarManager bar)
{
if (IsLanguage)
{
string key = String.Empty;
foreach (BarItem item in bar.Items)
{
key = String.Format("{0}.{1}", name, item.Name);
string val = GetString(key);
if (!String.IsNullOrEmpty(val))
{
item.Caption = val;
}
}
}
} /// <summary>
/// 应用控件,改变语言显示
/// /summary>
/// <param name="name"></param>
/// <param name="ctrls"></param>
public static void ApplyControls(string name, Control.ControlCollection ctrls)
{
if (IsLanguage)
{
foreach (Control ctrl in ctrls)
{
if (ctrl is MenuStrip) //MenuStrip StatusStrip
{
ApplyMenuStrip(name, (MenuStrip) ctrl);
}
else if (ctrl is StatusStrip)
{
ApplyStatusStrip(name, (StatusStrip) ctrl);
}
else if (ctrl is ListView)
{
ApplyListView(name, (ListView) ctrl);
}
else
{
ApplyControls(name, ctrl);
} if (ctrl.HasChildren)
{
ApplyControls(name, ctrl.Controls);
}
}
}
} internal static bool IsLanguage
{
get
{
if (File.Exists(_cnPath) && File.Exists(_enPath))
{
return true;
}
else
{
return false;
}
}
} private static void ApplyControls(string name, Control ctrl)
{
string key = String.Format("{0}.{1}", name, ctrl.Name);
string text = GetString(key);
if (!String.IsNullOrEmpty(text))
{
ctrl.Text = text;
}
} private static void ApplyMenuStrip(string name, MenuStrip menu)
{
foreach (ToolStripMenuItem item in menu.Items)
{
ApplyMenuItem(name, item);
}
} private static void ApplyMenuItem(string name, ToolStripMenuItem item)
{
string key = String.Format("{0}.{1}", name, item.Name);
string text = GetString(key);
if (!String.IsNullOrEmpty(text))
{
item.Text = text;
}
if (item.DropDownItems.Count > 0)
{
foreach (ToolStripMenuItem subItem in item.DropDownItems)
{
ApplyMenuItem(name, subItem);
}
}
} private static void ApplyStatusStrip(string name, StatusStrip status)
{
string key = String.Empty;
foreach (ToolStripItem item in status.Items)
{
key = String.Format("{0}.{1}", name, item.Name);
string val= GetString(key);
if (!String.IsNullOrEmpty(val))
{
item.Text = val;
}
}
} private static void ApplyListView(string name, ListView lv)
{
string key = String.Empty;
foreach (ColumnHeader header in lv.Columns)
{
key = String.Format("{0}.{1}", name, header.Tag == null ? "" : header.Tag.ToString());
string val = GetString(key);
if (!String.IsNullOrEmpty(val))
{
header.Text = val;
}
}
}
}

13.4     应用管理类

在软件启动时可以使用CultureMananger管理类,具体应用代码如下:

CultureMananger.LoadCulture();
CultureMananger.ApplyControls("MainForm",this.Controls);
string state=CultureMananger.GetString("State.Normal");

13.5     小结

这是一个小的工具组件,具有一定的通用性。

作者:唯笑志在

Email:504547114@qq.com

QQ:504547114

.NET开发技术联盟:54256083

文档下载:http://pan.baidu.com/s/1pJ7lZWf

官方网址:http://www.bmpj.net

[连载]《C#通讯(串口和网络)框架的设计与实现》- 13.中英文版本切换设计的更多相关文章

  1. [连载]《C#通讯(串口和网络)框架的设计与实现》- 0.前言

                                  目       录 前言 前言 刚参加工作,使用过VB.VC开发软件,随着C#的崛起,听说是C++++,公司决定以后开发软件使用C#,凭借在 ...

  2. 《连载 | 物联网框架ServerSuperIO教程》-4.如开发一套设备驱动,同时支持串口和网络通讯。附:将来支持Windows 10 IOT

    1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制. <连载 | 物联网框架Serve ...

  3. [连载]《C#通讯(串口和网络)框架的设计与实现》-1.通讯框架介绍

    [连载]<C#通讯(串口和网络)框架的设计与实现>- 0.前言 目       录 第一章           通讯框架介绍... 2 1.1           通讯的本质... 2 1 ...

  4. [连载]《C#通讯(串口和网络)框架的设计与实现》- 6.通讯控制器的设计

    目       录 第六章           通讯控制器的设计... 2 6.1           控制器接口... 2 6.2           串口控制器... 3 6.3          ...

  5. [连载]《C#通讯(串口和网络)框架的设计与实现》-2.框架的总体设计

    目       录 C#通讯(串口和网络)框架的设计与实现... 1 (SuperIO)- 框架的总体设计... 1 第二章           框架总体的设计... 2 2.1           ...

  6. [连载]《C#通讯(串口和网络)框架的设计与实现》- 5.串口和网络统一IO设计

    目       录 第五章           串口和网络统一IO设计... 2 5.1           统一IO接口... 2 5.1.1    串口IO.. 4 5.1.2    网络IO.. ...

  7. [连载]《C#通讯(串口和网络)框架的设计与实现》-4.设备驱动管理器的设计

    目       录 第四章           设备驱动管理器的设计... 2 4.1           接口定义... 2 4.2           设备容器... 7 4.3          ...

  8. [连载]《C#通讯(串口和网络)框架的设计与实现》-3.设备驱动的设计

    目       录 第三章           设备驱动的设计... 2 3.1           初始化设备... 4 3.2           运行设备接口设计... 4 3.3        ...

  9. [连载]《C#通讯(串口和网络)框架的设计与实现》- 7.外部接口的设计

    目       录 第七章           外部接口的设计... 2 7.1           插件接口... 2 7.2           图形显示接口... 3 7.3           ...

随机推荐

  1. 微信小程序开发日记——高仿知乎日报(中)

    本人对知乎日报是情有独钟,看我的博客和github就知道了,写了几个不同技术类型的知乎日报APP要做微信小程序首先要对html,css,js有一定的基础,还有对微信小程序的API也要非常熟悉 我将该教 ...

  2. BPM配置故事之案例14-数据字典与数据联动

    小明遇到了点麻烦,他昨天又收到了行政主管发来的邮件,要求把出差申请单改由H3 BPM进行,表单如下 行政主管的出差申请表 小明对表单进行了调整,设计出了一份适合在系统中使用的表单,但在"出差 ...

  3. Windows 上安装 Jekyll.

    Jekyll是一个静态网站生成工具.它允许用户使用HTML.Markdown或Textile来建立静态页面,然后通过模板引擎Liquid(Liquid Templating Engine)来运行. 原 ...

  4. Webpack+React+ES6开发模式入门指南

    React无疑是今年最火的前端框架,github上的star直逼30,000,基于React的React Native的star也直逼20,000.有了React,组件化似乎不再步履蹒跚,有了Reac ...

  5. 死去活来,而不变质:Domain Model(领域模型) 和 EntityFramework 如何正确进行对象关系映射?

    写在前面 阅读目录: 设计误区 数据库已死 枚举映射 关联映射 后记 在上一篇<一缕阳光:DDD(领域驱动设计)应对具体业务场景,如何聚焦 Domain Model(领域模型)?>博文中, ...

  6. 谈一谈Http Request 与 Http Response

    写在前面的话:今天来总结一下http相关的request和response,就从以下几个问题入手吧. ======正文开始======== 1.什么是HTTP Request 与HTTP Respon ...

  7. Entity Framework 6 Recipes 2nd Edition(13-9)译 -> 避免Include

    问题 你想不用Include()方法,立即加载一下相关的集合,并想通过EF的CodeFirst方式实现. 解决方案 假设你有一个如Figure 13-14所示的模型: Figure 13-14. A ...

  8. 关于python中的flush问题

    今天写了一个控制台下的进度条小程序,遇到了关于flush的问题,在这里小小的总结一下. 首先在要flush的字符串必须后边加上'\r'(回车) 将数据送入缓冲区这样在新打印字符的时候就可以将原来的显示 ...

  9. C#中,switch case语句中多个值匹配一个代码块的写法

    switch (num) { case 1: Response.Write("1"); break; case 2: case 3: Response.Write("2| ...

  10. python的print函数的格式化输出

    使用print函数的时候,可以像C一样格式化输出,同时还支持参数化输出 print('%s' % ("CooMark")) print('整数|%d|' % (123)) prin ...