上篇文章我们介绍了

.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. WPF 实际国际化多语言界面

    前段时候写了一个WPF多语言界面处理,个人感觉还行,分享给大家.使用合并字典,静态绑定,动态绑定.样式等东西 效果图 定义一个实体类LanguageModel,实际INotifyPropertyCha ...

  2. yii2 api接口 实现国际化多语言设置

    1) 在 /config/main.php 下添加如下代码: 'components' => [ 'language' => 'zh-CN', 'i18n' => [ 'transl ...

  3. Linux下获得系统时间的C语言实现

    Linux下获得系统时间的C语言的实现方法 #include<time.h> //C语言的头文件#include<stdio.h> //C语言的I/O   int main() ...

  4. iOS 国际化多语言设置 xcode7

    iOS 国际化多语言设置 方式一: 1. 在storyboard中创建好UI,然后在 project 里面  Localizables 栏目里面,添加你需要的语言:默认是Englist; 比如这里我添 ...

  5. Form_Form Builder国际化多语言开发(案例)

    2014-05-06 Created By BaoXinjian

  6. 更好用的excel国际化多语言导出

    不知道大家在开发中有没有遇到过『excel导出』的需求,反正我最近写了不少这种功能,刚开始利用poi,一行行的手动塞数据,生成excel,而且还有国际化需求,比如:标题栏有一列,用户切换成" ...

  7. [Spring]Spring Mvc实现国际化/多语言

    1.添加多语言文件*.properties F64_en_EN.properties详情如下: F60_G00_M100=Please select data. F60_G00_M101=Are yo ...

  8. iOS 学习笔记六 【APP中的文字和APP名字的国际化多语言处理】

    今天为新手解决下APP中的文字和APP名字的国际化多语言处理, 不多说了,直接上步骤: 1.打开你的项目,单机project名字,选中project,直接看图吧: 2.创建Localizable.st ...

  9. php gettext方式实现UTF-8国际化多语言(i18n)

    php gettext方式实现UTF-8国际化多语言(i18n) 一.总结 一句话总结: 二.php gettext方式实现UTF-8国际化多语言(i18n) 近 来随着i18n(国际化)的逐渐标准化 ...

  10. 系统区域设置 本地语言的支持依赖于 /etc/locale.conf,/etc/locale.conf 包含不少于此相关的环境变量

    https://linux.cn/lfs/LFS-BOOK-7.7-systemd/chapter07/locale.html 7.7. 系统区域设置 本地语言的支持依赖于 /etc/locale.c ...

随机推荐

  1. Spring boot jar包解压后重新压缩命令

    进入解压的目录/demo,运行 jar cvfM0 demo.jar * 压缩后的项目即可运行 参考:https://www.cnblogs.com/liyanbin/p/6088458.html

  2. 【2020NOI.AC省选模拟#6】A. zyb的监控计划

    题目链接 原题解: 考虑我们需要的信息:子树里最浅的一个能向上的点是谁?子树里最深的一个没被覆盖的深度是多少? 我们记录一下$f_{i,a,b}$表示上面两个信息为$a$和$b$的时候,最少要花费的代 ...

  3. matlab 求解 f(x)=x(x+1)(x+2)(x+3)(x+4)...(x+n-2)(x+n-1)(x+n)的导数;

    matlab 求解 f(x)=x(x+1)(x+2)(x+3)(x+4)...(x+n-2)(x+n-1)(x+n)的导数; matlab diff() 问题的提出 问题 代码求解 clc; clea ...

  4. 天龙八部<三联版>二

    段誉来到无锡,遇上乔峰,斗气与乔峰比酒,乔峰误以为段誉是慕容复,段誉被乔峰豪气所折服,像乔峰坦言,乔峰不但不怒,反而对段誉很认可,二人结拜 段誉跟随乔峰来到丐帮,包不同和三女因为慕容复而和丐帮发生争执 ...

  5. Erlang Mnesia数据库迁移方法

    本文参考https://blog.csdn.net/yangzm/article/details/51686249 需求 因为一些原因,需要把一个Mnesia节点的数据库搬迁到另一个节点,然后弃用原来 ...

  6. Filbeat采集nginx-ingress日志

    一.创建configmap配置文件 注:filebeat6以上版本需要将prospectors改为inputs,paths下指定的nginx-ingress日志路径匹配模式以及hosts指定的kafk ...

  7. springboot 日志处理

    引言 springboot框架集成logback日志 logback是由log4j创始人设计的又一个开源日志组件.目前,logback分为三个模块:logback-core,logback-class ...

  8. C# 锁汇总

    一.前言 本文章汇总c#中常见的锁,基本都列出了该锁在微软官网的文章,一些不常用的锁也可以参考微软文章左侧的列表,方便温习回顾. 二.锁的分类 2.1.用户模式锁 1.volatile 关键字 vol ...

  9. 【Unity】关于VS条件编译符号

    写在前面 起因:我在回顾LuaFramework_UGUI(作者Jarjin Lee)代码时,看到了C#代码中的条件编译符号,比如下图的ASYNC_MODE.虽然字面上知道是什么意思,但我对VS的条件 ...

  10. Android Studio查看指定APP日志

    1.启动Android Studio 2.View->Tool Windows->Terminal   3.在底部输入monitor 4.Android Devices Monitor新窗 ...