草庐IT

.NET应用系统的国际化-多语言翻译服务

I love .net 2023-03-28 原文

上篇文章我们介绍了

.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

有关.NET应用系统的国际化-多语言翻译服务的更多相关文章

  1. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  2. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  3. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  4. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  5. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  6. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  7. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  8. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  9. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  10. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

随机推荐