.NET應用系統的國際化-多語言翻譯服務

来源:https://www.cnblogs.com/tianqing/archive/2023/03/20/17232559.html
-Advertisement-
Play Games

上篇文章我們介紹了 .NET應用系統的國際化-基於Roslyn抽取詞條、更新代碼 系統國際化改造整體設計思路如下: 提供一個工具,識別前後端代碼中的中文,形成多語言詞條,按語言、界面、模塊統一管理多有的多語言詞條 提供一個翻譯服務,批量翻譯多語言詞條 提供一個詞條服務,支持後端代碼在運行時根據用戶登 ...


上篇文章我們介紹了

.NET應用系統的國際化-基於Roslyn抽取詞條、更新代碼

系統國際化改造整體設計思路如下:

  1. 提供一個工具,識別前後端代碼中的中文,形成多語言詞條,按語言、界面、模塊統一管理多有的多語言詞條
  2. 提供一個翻譯服務,批量翻譯多語言詞條
  3. 提供一個詞條服務,支持後端代碼在運行時根據用戶登錄的語言,動態獲取對應的多語言文本
  4. 提供前端多語言JS生成服務,按界面動態生成對應的多語言JS文件,方便前端VUE文件使用。
  5. 提供代碼替換工具,將VUE前端代碼中的中文替換為$t("詞條ID"),後端代碼中的中文替換為TermService.Current.GetText("詞條ID")

本篇文章我們重點和大家分享多語言翻譯服務的設計和實現。

一、業務背景

通過上一篇文章,我們把sln解決方案中各個Project下的中文文本,識別成大量的多語言詞條。

這些多語言詞條臨時存儲在資料庫中,我們要對這個臨時結果集,通過多語言翻譯服務,按支持的語言,翻譯成多語言詞條。

對應的類圖設計:

 

 

 對應的詞條管理界面:

 

 

 因此我們需要一個多語言詞條翻譯服務,實現詞條的批量、快速機器翻譯。

二、多語言詞條翻譯服務

首先,抽象一個翻譯介面II18NTermTranslateService

/// <summary>
/// 詞條翻譯服務介面
/// </summary>
public interface II18NTermTranslateService
{
     string Translate(string text, string language);
}

設計一個翻譯服務提供者類,通過Facade模式,對外統一提供翻譯服務TranslateServiceProvider

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using I18N.SPI;

namespace I18N.Translation
{
    /// <summary>
    /// 翻譯服務提供者
    /// </summary>
    public class TranslateServiceProvider
    {
        public static II18NTermTranslateService GetTranslateService(Translater translater)
        {
            switch (translater)
            {
                case Translater.Youdao:
                default:
                    return new YoudaoTranslateService();
                case Translater.Baidu:
                    return new BaiduTranslateService();
                case Translater.Google:
                    return new GoogleTranslateService();
                case Translater.Azure:
                    return new AzureTranslateService();
            }
        }

        public static II18NTermTranslateService GetYoudaoTranslateService()
        {
            return new YoudaoTranslateService();
        }

        public static II18NTermTranslateService GetGoogleTranslateService()
        {
            return new GoogleTranslateService();
        }

        public static II18NTermTranslateService GetBaiduTranslateService()
        {
            return new BaiduTranslateService();
        }

        public static II18NTermTranslateService GetAzureTranslateService()
        {
            return new AzureTranslateService();
        }
    }
}

 這裡的Translater是個枚舉

public enum Translater
    {
        Youdao,
        Baidu,
        Google,
        Azure
    }

三、多語言詞條翻譯服務-Azure翻譯服務

這裡我們使用Azure認知服務中的服務服務,實現上面抽象好的翻譯介面

using Newtonsoft.Json;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Http;
using System.Text;
using I18N.SPI;

namespace I18N.Translation
{
    /// <summary>
    /// Azure翻譯服務
    /// </summary>
    /// <remarks>
    /// https://learn.microsoft.com/zh-cn/azure/cognitive-services/translator/text-translation-overview
    /// </remarks>
    public class AzureTranslateService : II18NTermTranslateService
    {
        private readonly string _endpoint = "https://api.cognitive.microsofttranslator.com";
        private readonly string _key = "XXXXXXXXXXXXXX";
        public string Translate(string text, string language)
        {
            return Post(text, language);
        }

        private string Post(string text, string language)
        {
            using (var client = new HttpClient())
            {
                using (var request = new HttpRequestMessage())
                {
                    var url = $"/translate?api-version=3.0&to={language}";
                    request.Method = HttpMethod.Post;
                    request.RequestUri = new Uri($"{_endpoint}{url}");
                    object[] body = { new { Text = text } };
                    var requestBody = JsonConvert.SerializeObject(body);
                    request.Content = new StringContent(requestBody, Encoding.UTF8, "application/json");
                    request.Headers.Add("Ocp-Apim-Subscription-Key", _key);

                    var response = client.SendAsync(request).ConfigureAwait(false).GetAwaiter().GetResult();
                    if (!response.IsSuccessStatusCode)
                    {
                        return null;
                    }
                    string result = response.Content.ReadAsStringAsync().Result;
                    var translationResults = JsonConvert.DeserializeObject<TranslationResult[]>(result);
                    if (translationResults.Length > 0)
                    {
                        return translationResults[0].Translations.FirstOrDefault()?.Text;
                    }
                }
            }
            return null;
        }
    }
}

這裡用到了幾個參數類

namespace I18N.Translation
{
    public class TranslationResult
    {
        public DetectedLanguage DetectedLanguage { get; set; }
        public Translation[] Translations { get; set; }
    }

    public class DetectedLanguage
    {
        public string Language { get; set; }
        public float Score { get; set; }
    }
    public class Translation
    {
        public string Text { get; set; }
        public string To { get; set; }
    }
}

四、多語言詞條翻譯服務-有道雲翻譯服務

這裡我們同時實現了有道雲翻譯服務

using System;
using System.Linq;
using System.Collections.Generic;
using System.Net.Http;
using System.Web;
using I18N.SPI;

namespace I18N.Translation
{
    /// <summary>
    /// 有道雲翻譯服務
    /// </summary>
    /// <remarks>
    /// https://ai.youdao.com/DOCSIRMA/html/%E8%87%AA%E7%84%B6%E8%AF%AD%E8%A8%80%E7%BF%BB%E8%AF%91/API%E6%96%87%E6%A1%A3/%E6%96%87%E6%9C%AC%E7%BF%BB%E8%AF%91%E6%9C%8D%E5%8A%A1/%E6%96%87%E6%9C%AC%E7%BF%BB%E8%AF%91%E6%9C%8D%E5%8A%A1-API%E6%96%87%E6%A1%A3.html
    /// </remarks>
    public class YoudaoTranslateService : II18NTermTranslateService
    {
        private string appKey = "XXXXXX";
        private string appKeyMY = "XXXXXXXXX";

        public string Translate(string text, string language)
        {
            return Post(text, language);
        }

        public void Translate(List<I18NTerm> terms, string language)
        { }

        private string Post(string text, string language)
        {
            var salt = ToUnixTime(DateTime.Now);
            var sign = Encryptor.MD5Hash(appKey + text + salt + appKeyMY).ToUpper();

            switch (language.ToLower())
            {
                case "zh-cn": language = "zh-CHS"; break;
                case "en-us": language = "EN"; break;
            }

            HttpClient client = new HttpClient();

            var encodedText = System.Uri.EscapeUriString(text);
            var url = @"https://openapi.youdao.com/api?q=" + encodedText + "&from=auto&to=" + language + "&appKey=" + appKey + "&salt=" + salt + "&sign=" + sign;            
            var result = client.GetStringAsync(url).Result;

            //{"returnPhrase":["系統"],"query":"系統","errorCode":"0","l":"zh-CHS2en","tSpeakUrl":"https://openapi.youdao.com/ttsapi?q=system&langType=en&sign=F1945F1CB2D0AEEE40B1277E6C871770&salt=1665580681639&voice=4&format=mp3&appKey=48045ce9f1d5f934&ttsVoiceStrict=false","web":[{"value":["System","lineage","Systematic problem-solving","Windows XP"],"key":"系統"},{"value":["Operating System","OS","Linux"],"key":"操作系統"},{"value":["Domain Name System","Domain Name Server","Domain System"],"key":"功能變數名稱系統"}],"requestId":"cf134fc6-812b-49ab-a97a-85e56e6697cd","translation":["system"],"dict":{"url":"yddict://m.youdao.com/dict?le=eng&q=%E7%B3%BB%E7%BB%9F"},"webdict":{"url":"http://mobile.youdao.com/dict?le=eng&q=%E7%B3%BB%E7%BB%9F"},"basic":{"phonetic":"xì tǒng","explains":["system"]},"isWord":true,"speakUrl":"https://openapi.youdao.com/ttsapi?q=%E7%B3%BB%E7%BB%9F&langType=zh-CHS&sign=DF2CDF4E306FC8C4F8E224C6E7436B26&salt=1665580681639&voice=4&format=mp3&appKey=48045ce9f1d5f934&ttsVoiceStrict=false"}
            var dataResult = Newtonsoft.Json.JsonConvert.DeserializeObject<YoudaoResult>(result);
            if (dataResult != null)
            {
                return dataResult.translation.FirstOrDefault();
            }

            return null;
        }

        private long ToUnixTime(DateTime dateTime)
        {
            var start = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);

            return Convert.ToInt64((dateTime.ToUniversalTime() - start).TotalMilliseconds);
        }
    }
}

有道雲的翻譯HttpAPI涉及到了幾個參數類

  public class YoudaoResult
    {
        /// <summary>
        /// 錯誤返回碼
        /// </summary>
        public string errorCode { get; set; }

        /// <summary>
        /// 源語言和目標語言
        /// </summary>
        public string l { get; set; }

        /// <summary>
        /// 源語言
        /// </summary>
        public string query { get; set; }

        /// <summary>
        /// 源語言
        /// </summary>
        public List<string> translation { get; set; }
    }

 

以上是和大家分享多語言翻譯服務的設計和實現。

 

周國慶

2023/3/19


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文已收錄至Github,推薦閱讀 👉 Java隨想錄 微信公眾號:Java隨想錄 摘要 空指針異常(NullPointerException)可以說是Java程式員最容易遇到的問題了,影響說大不大,說小也不小。為瞭解決這個問題,Java 8 版本中推出了 Optional 類。Optional ...
  • _pycache__文件夾可以看作該文件夾下文件已被python接管或者說編譯過。 在第一次執行代碼的時候,Python解釋器已經把編譯的位元組碼放在__pycache__文件夾中,這樣以後再次運行的話,如果被調用的模塊未發生改變,那就直接跳過編譯這一步,直接去__pycache__文件夾中去運行相關 ...
  • 目前由於功能比較簡單,沒有進行前後端拆分,主要技術點:SpringBoot、WebSocket、webjars、Vue、Element-UI、axios、jszip,核心壓縮庫是 Thumbnailator 提供壓縮支持。源代碼已經在 碼雲 star 公開。歡迎各位大佬批評。 ...
  • 解讀 Servlet 源碼:GenericServlet,ServletConfig,ServletContext 每博一文案 人活著,就得隨時準備經受磨難。他已經看過一些書,知道不論是普通人還是了不起的人,都要在自己的一生中經歷許多磨難。磨難使人堅強。 人和社會、一切鬥爭的總結局也許都是中庸而已。 ...
  • 在剛接觸後臺線程的時候,覺得線程神秘且高深,並且時常有先輩們千叮萬囑:能不用的時候,儘量不要用,千萬不要濫用線程,否則會發生預料不到的結果。在接觸線程一段時間後,感覺線程也不過如此,輕而易舉的就可以創建,所以逐漸大膽起來,項目里隨處可見的都是Task,Thread,async,await等內容。在大... ...
  • 一周一周來算,註,如果本月1號是星期日,則這一月共有六周,即六行,所以乾脆每月都輸出6周就行了,然後再算第一周里的本月一號是周幾,第一周里本月又占了幾天。。。呃。。。反正很亂,想了一上午,現在寫出來了倒不知道如何解釋了。。算了。。直接上代碼。。 //日曆界面 public IActionResult ...
  • 調用Win32 API(優先順序最高,全局監聽, 支持最小化失焦等情況) 那麼,假如我要在一個WPF程式監聽CTRL+5按鍵,首先在主視窗程式添加以下代碼: /// <summary> /// CTRL+5事件Id /// </summary> private const int Ctrl5KeyEv ...
  • 1. 編碼原則 1.1. SOLID原則 1.1.1. 單一職責原則(Single Respon-sibility Principle) 1.1.1.1. 類和方法應當僅具備單一職責。所有組合為單一職責的元素應當組合在一起併進行封裝。 1.1.2. 開閉原則(Open-Closed Principl ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...