溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何使用gettext技術(shù)為ASP.NET網(wǎng)站實現(xiàn)國際化支持

發(fā)布時間:2021-11-24 13:39:18 來源:億速云 閱讀:133 作者:柒染 欄目:編程語言

這篇文章將為大家詳細(xì)講解有關(guān)如何使用gettext技術(shù)為ASP.NET網(wǎng)站實現(xiàn)國際化支持,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。

不知道有多少人對這個題目感興趣,因為最近在做一個網(wǎng)站玩玩,有點閑心給網(wǎng)站加了國際化支持。雖然ASP.NET已經(jīng)有ResourceManager這個類,并且有標(biāo)簽實現(xiàn)國際化的支持了,但是它的問題是,ResourceManager對每一個需要翻譯的句子都要求有一個鍵(Key):

1. 要先創(chuàng)建一個.resx文件,在Visual Studio里,有一個工具編輯這個.resx文件。

2. 對每一個需要翻譯的句子,添加一個鍵值對。

3. 然后在代碼里,使用ResourceManager或者<%#這個標(biāo)簽,通過定義好的鍵來告訴ASP.NET在運行的時候查找正確的翻譯文本。

太麻煩了,不知道大家有什么其它好的方法,我使用的方法是從unix gettext那邊借用過來的理念。

理念

Gettext的理念很簡單,文本翻譯嗎,說白了就是把一句話翻譯成另外一句話嘛,這個要翻譯的句子,本身就可以當(dāng)做檢索要用的關(guān)鍵字,何必要再新建一個另外的關(guān)鍵字呢?gettext的方式很簡單:

1. 在源代碼里,你可以編寫一個特殊的函數(shù)執(zhí)行翻譯,這個函數(shù)只接受一個參數(shù),就是要翻譯的文本。

2. 使用一個輔助程序xgettext掃描源代碼的文本,將所有待翻譯的文本都找出來,保存到一個文件里,一般來說,這個文件叫做po文件。

3. 因為ASP.NET程序不支持po文件,再使用一個輔助程序msgfmt將po文件轉(zhuǎn)換成ASP.NET支持的.resources文件。這個方法的優(yōu)點在于:

1. 你在編寫程序的時候,不用為需要翻譯的句子,定義一個新的關(guān)鍵字&mdash;&mdash;這個關(guān)鍵字一般都比較難理解,也不好取名。在維護(hù)代碼的時候很麻煩&mdash;&mdash;因為你需要不停地在.resx編輯器和cs文件之間切換。

2. 不知道怎么搞的,很難找到可以編輯.resx文件的工具,而gettext生成的po文件是普通的文本文件,而且格式非常簡單。這樣在翻譯的時候,就很方便了。

做法

比如寫了一個ASP.NET MVC程序,當(dāng)然窗體(Web Form)形式的程序理念也是一樣的,

1. 寫一個控制器和視圖頁的基類,里面都有一個執(zhí)行翻譯的函數(shù)T:

public class G18nController : Controller  {  public CultureInfo Culture { get; set; }  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 { get; set; }  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. 在代碼里,針對每個要翻譯的句子,直接調(diào)用這個T函數(shù)好了:

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

3. 程序?qū)懞煤螅_始翻譯,調(diào)用gettext程序?qū)⑺幸g的句子找出來,保存到指定的po文件里。可以在http://gnuwin32.sourceforge.net/packages/gettext.htm這個網(wǎng)頁下載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文件格式其實非常簡單易懂:

#: C:\workspace\Views\Role\Edit.cshtml:9  msgid "管理用戶組" msgstr "" #: C:\workspace\Views\Role\Edit.cshtml:23  msgid "用戶組[{0}]的權(quán)限" msgstr ""

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

5. 完成翻譯后,使用一個輔助程序msgfmt將翻譯好的po文件轉(zhuǎn)換成ASP.NET支持的格式。因為原始的gettext程序包里的msgfmt.exe好像不能生成ASP.NET識別的.resources文件,所以 我也寫了一個msgfmt程序完成這個工作&mdash;&mdash;在本文的附件里可以下載到,命令格式是:

Msgfmt -o 輸出的resource文件路徑 -i 輸入的po文件路徑

例如:

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

注意:輸出的resource文件名,必須與你在***步里,使用HttpContext.GetGlobalResourceObject函數(shù)的***個參數(shù)相同。

6. 我寫了一個小的批處理,將3、4、5步結(jié)合在一起執(zhí)行:

pushd src  del /F source.lst  dir /s /b src\*.cs >> source.lst  dir /s /b src\*.cshtml >> source.lst  tools\zgettext\zgettext\bin\Debug\zgettext.exe -k T -f source.lst -o glob\website.po  tools\zgettext\msgfmt\bin\Debug\msgfmt.exe -o src\App_GlobalResources\website.resources -i glob\ website.po  popd

關(guān)于如何使用gettext技術(shù)為ASP.NET網(wǎng)站實現(xiàn)國際化支持就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI