使用gettext技术为ASP.NET网站实现国际化支持

简介:

不知道有多少人对这个题目感兴趣,因为最近在做一个网站玩玩,有点闲心给网站加了国际化支持。虽然ASP.NET已经有ResourceManager这个类,并且有标签实现国际化的支持了,但是它的问题是,ResourceManager对每一个需要翻译的句子都要求有一个键(Key):

1.       要先创建一个.resx文件,在Visual Studio里,有一个工具编辑这个.resx文件。

2.       对每一个需要翻译的句子,添加一个键值对。

3.       然后在代码里,使用ResourceManager或者<%#这个标签,通过定义好的键来告诉ASP.NET在运行的时候查找正确的翻译文本。

太麻烦了,不知道大家有什么其它好的方法,我使用的方法是从unix gettext那边借用过来的理念。

理念

Gettext的理念很简单,文本翻译吗,说白了就是把一句话翻译成另外一句话嘛,这个要翻译的句子,本身就可以当做检索要用的关键字,何必要再新建一个另外的关键字呢?gettext的方式很简单:

1.       在源代码里,你可以编写一个特殊的函数执行翻译,这个函数只接受一个参数,就是要翻译的文本。

2.       使用一个辅助程序xgettext扫描源代码的文本,将所有待翻译的文本都找出来,保存到一个文件里,一般来说,这个文件叫做po文件。

3.       因为ASP.NET程序不支持po文件,再使用一个辅助程序msgfmtpo文件转换成ASP.NET支持的.resources文件。

这个方法的优点在于:

1.       你在编写程序的时候,不用为需要翻译的句子,定义一个新的关键字——这个关键字一般都比较难理解,也不好取名。在维护代码的时候很麻烦——因为你需要不停地在.resx编辑器和cs文件之间切换。

2.       不知道怎么搞的,很难找到可以编辑.resx文件的工具,而gettext生成的po文件是普通的文本文件,而且格式非常简单。这样在翻译的时候,就很方便了。

做法

比如写了一个ASP.NET MVC程序,当然窗体(Web Form)形式的程序理念也是一样的,

1.       写一个控制器和视图页的基类,里面都有一个执行翻译的函数T

    public class G18nController : Controller

    {

        public CultureInfo Culture { getset; }

 

        public string T(string message)

        {

            var obj = HttpContext.GetGlobalResourceObject("website", message, Culture);

            var translated = obj == null ? null : obj.ToString();

            if (string.IsNullOrEmpty(translated))

                return message;

            else

                return translated;

        }

    }

 

    public abstract class G18nWebViewPage<U> : WebViewPage<U>

    {

        public CultureInfo Culture { getset; }

 

        public string T(string message)

        {

            var obj = HttpContext.GetGlobalResourceObject("website", message, Culture);

            var translated = obj == null ? null : obj.ToString();

            if (string.IsNullOrEmpty(translated))

                return message;

            else

                return translated;

        }

    }

 

上面的Culture属性,可以从Request.Headers["Accept-Language"]属性取得。

 

2.       在代码里,针对每个要翻译的句子,直接调用这个T函数好了:

throw new ArgumentException(string.Format(T("找不到ID{0}的项目!"), id));

 

3.       程序写好后,要开始翻译,调用gettext程序将所有要翻译的句子找出来,保存到指定的po文件里。可以在http://gnuwin32.sourceforge.net/packages/gettext.htm这个网页下载gettext

 

但是悲剧的是,gettext好像要求主语言是英文,对中文字符串支持的不是很好。所以我就用C#自己写了一个gettext,你可以在本文的附件里下载它,命令的格式是:

Zgettext -k T -i 源代码路径名 -o 输出的po文件名

Zgettext -k T -f 源代码路径列表文件 -o 输出的po文件名

 

比如:

Zgettext -k T -i AccountController.cs -o test.po

 

4.       生成的po文件格式其实非常简单易懂:

1.  #: C:\workspace\Views\Role\Edit.cshtml:9

2.  msgid "管理用户组"

3.  msgstr ""

4.  

 

5.  #: C:\workspace\Views\Role\Edit.cshtml:23

6.  msgid "用户组[{0}]的权限"

7.  msgstr ""

       

        Msgid就是要翻译的句子,msgstr就是翻译好的句子。

5.       完成翻译后,使用一个辅助程序msgfmt将翻译好的po文件转换成ASP.NET支持的格式。因为原始的gettext程序包里的msgfmt.exe好像不能生成ASP.NET识别的.resources文件,所以 我也写了一个msgfmt程序完成这个工作——在本文的附件里可以下载到,命令格式是:

 

Msgfmt -o 输出的resource文件路径 -i 输入的po文件路径

 

例如:

Msgfmt -o website.en-US.resources -i website.po

 

注意:输出的resource文件名,必须与你在第一步里,使用HttpContext.GetGlobalResourceObject函数的第一个参数相同。

 

6.       我写了一个小的批处理,将345步结合在一起执行:

1.  pushd src

2.  del /F source.lst

3.  dir /s /b src\*.cs >> source.lst

4.  dir /s /b src\*.cshtml >> source.lst

5.  tools\zgettext\zgettext\bin\Debug\zgettext.exe -k T -f source.lst -o glob\website.po

6.  tools\zgettext\msgfmt\bin\Debug\msgfmt.exe -o src\App_GlobalResources\website.resources -i glob\ website.po

7.       popd

 

希望对你有点帮助。




本文转自 donjuan 博客园博客,原文链接:  http://www.cnblogs.com/killmyday/archive/2011/04/21/2023816.html   ,如需转载请自行联系原作者
相关文章
|
7月前
|
存储 C# 数据库
.NET开源的在Windows上统计软件使用时长和网站浏览时长工具 - Tai
.NET开源的在Windows上统计软件使用时长和网站浏览时长工具 - Tai
|
3月前
|
Java PHP 开发工具
支付宝记---电脑网站支付(.NET)
支付宝记---电脑网站支付(.NET)
40 1
|
Docker 容器
.NET Core 网站项目在Docker中运行无法访问问题处理
.NET Core 网站项目在Docker中运行无法访问问题处理
264 0
|
10月前
|
开发框架 供应链 前端开发
net基于asp.net的社区团购网站
社区团购系统依托社区团购系统和社区门店,是现在的一个重大市场和发展方向,通过研究企业在社区团购系统环境下的营销模式创新,对于普通的零售业和传统社区团购系统的转型发展具有重要的理论意义。随着互联网行业的发展,人们的生活方式发生着重大变化,人们越来越倾向于网络购物,这对传统企业来说如何把客户留下是一个重大挑战。就现在而言,由于社区团购的竞争已经进入最紧张激烈的阶段,有些团购平台甚至已经彼此之间打起了价格战,其中不乏有平台因为利润变少或资金链断裂而半途败亡。企业在实际的商业活动中,往往会面临许多等待优化的问题。因此,要在竞争激烈的市场中拔得头筹,必须重视提升对新商业模式的全面认知,科学于实际贴合的分
|
11月前
|
开发框架 安全 .NET
教你如何在WINDOWS Server2003上部署一个Asp.Net的网站
教你如何在WINDOWS Server2003上部署一个Asp.Net的网站
302 0
|
开发框架 JSON 前端开发
6.3 ASP.NET Core Web API技术选择
ASP.NET Core Web API技术选择
6.3 ASP.NET Core Web API技术选择
|
前端开发 NoSQL 关系型数据库
一个基于.Net Core开源个人博客网站
一个基于.Net 6开发的个人博客网站,功能完备。非常使用用来搭建个人博客、资源分享。
129 0
一个基于.Net Core开源个人博客网站
|
开发框架 .NET 应用服务中间件
|
开发框架 .NET 应用服务中间件
使用Nginx对ASP.NETCore网站或D ocker等进行反向代理,宝塔面板对ASP.NET Core 反向代理
使用Nginx对ASP.NETCore网站或D ocker等进行反向代理,宝塔面板对ASP.NET Core 反向代理
351 0
|
开发框架 .NET C#
ASP.NET Core 返回文件、用户 下载文件,从网站下载文件,动态下载文件
ASP.NET Core 返回文件、用户 下载文件,从网站下载文件,动态下载文件
497 0