[连载]《C#通讯(串口和网络)框架的设计与实现》- 13.中英文版本切换设计
目 录
第十三章 中英文版本切换设计... 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 语言管理类
- 定义一个词条对应的可序列化的类。代码如下:
[Serializable]
public class CultureItem
{
/// <summary>
/// 控件的级联ID,中间用"."分隔
/// </summary>
[XmlAttribute]
public string Key { set; get; } /// <summary>
/// 中文或英文描述
/// </summary>
[XmlAttribute]
public string Value { set; get; }
}
- 定义一个设置语言属性的枚举。代码如下:
public enum CultureLanguage
{
[EnumDescription("中文")]
Chinese,
[EnumDescription("英文")]
English
}
- 开发一个语言管理类库,本质上是根据语言配置文件对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.中英文版本切换设计的更多相关文章
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 0.前言
目 录 前言 前言 刚参加工作,使用过VB.VC开发软件,随着C#的崛起,听说是C++++,公司决定以后开发软件使用C#,凭借在 ...
- 《连载 | 物联网框架ServerSuperIO教程》-4.如开发一套设备驱动,同时支持串口和网络通讯。附:将来支持Windows 10 IOT
1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制. <连载 | 物联网框架Serve ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-1.通讯框架介绍
[连载]<C#通讯(串口和网络)框架的设计与实现>- 0.前言 目 录 第一章 通讯框架介绍... 2 1.1 通讯的本质... 2 1 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 6.通讯控制器的设计
目 录 第六章 通讯控制器的设计... 2 6.1 控制器接口... 2 6.2 串口控制器... 3 6.3 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-2.框架的总体设计
目 录 C#通讯(串口和网络)框架的设计与实现... 1 (SuperIO)- 框架的总体设计... 1 第二章 框架总体的设计... 2 2.1 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 5.串口和网络统一IO设计
目 录 第五章 串口和网络统一IO设计... 2 5.1 统一IO接口... 2 5.1.1 串口IO.. 4 5.1.2 网络IO.. ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-4.设备驱动管理器的设计
目 录 第四章 设备驱动管理器的设计... 2 4.1 接口定义... 2 4.2 设备容器... 7 4.3 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-3.设备驱动的设计
目 录 第三章 设备驱动的设计... 2 3.1 初始化设备... 4 3.2 运行设备接口设计... 4 3.3 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 7.外部接口的设计
目 录 第七章 外部接口的设计... 2 7.1 插件接口... 2 7.2 图形显示接口... 3 7.3 ...
随机推荐
- 微信小程序开发日记——高仿知乎日报(中)
本人对知乎日报是情有独钟,看我的博客和github就知道了,写了几个不同技术类型的知乎日报APP要做微信小程序首先要对html,css,js有一定的基础,还有对微信小程序的API也要非常熟悉 我将该教 ...
- BPM配置故事之案例14-数据字典与数据联动
小明遇到了点麻烦,他昨天又收到了行政主管发来的邮件,要求把出差申请单改由H3 BPM进行,表单如下 行政主管的出差申请表 小明对表单进行了调整,设计出了一份适合在系统中使用的表单,但在"出差 ...
- Windows 上安装 Jekyll.
Jekyll是一个静态网站生成工具.它允许用户使用HTML.Markdown或Textile来建立静态页面,然后通过模板引擎Liquid(Liquid Templating Engine)来运行. 原 ...
- Webpack+React+ES6开发模式入门指南
React无疑是今年最火的前端框架,github上的star直逼30,000,基于React的React Native的star也直逼20,000.有了React,组件化似乎不再步履蹒跚,有了Reac ...
- 死去活来,而不变质:Domain Model(领域模型) 和 EntityFramework 如何正确进行对象关系映射?
写在前面 阅读目录: 设计误区 数据库已死 枚举映射 关联映射 后记 在上一篇<一缕阳光:DDD(领域驱动设计)应对具体业务场景,如何聚焦 Domain Model(领域模型)?>博文中, ...
- 谈一谈Http Request 与 Http Response
写在前面的话:今天来总结一下http相关的request和response,就从以下几个问题入手吧. ======正文开始======== 1.什么是HTTP Request 与HTTP Respon ...
- Entity Framework 6 Recipes 2nd Edition(13-9)译 -> 避免Include
问题 你想不用Include()方法,立即加载一下相关的集合,并想通过EF的CodeFirst方式实现. 解决方案 假设你有一个如Figure 13-14所示的模型: Figure 13-14. A ...
- 关于python中的flush问题
今天写了一个控制台下的进度条小程序,遇到了关于flush的问题,在这里小小的总结一下. 首先在要flush的字符串必须后边加上'\r'(回车) 将数据送入缓冲区这样在新打印字符的时候就可以将原来的显示 ...
- C#中,switch case语句中多个值匹配一个代码块的写法
switch (num) { case 1: Response.Write("1"); break; case 2: case 3: Response.Write("2| ...
- python的print函数的格式化输出
使用print函数的时候,可以像C一样格式化输出,同时还支持参数化输出 print('%s' % ("CooMark")) print('整数|%d|' % (123)) prin ...