准备工作
halcon示例程序的描述部分一直是英文的,看起来很不方便。我决定汉化一下halcon示例程序的描述,准备工作如下:
拿到halcon的例程描述文件index_examples_en_us.xml,我的在c:\program files\mvtec\halcon-18.11-progress\help,其它版本类似。在注册拿到自己的id和密钥,开通通用翻译api服务。
注:百度官方有提供通用翻译api的使用demo,不过比较简单,c# demo下载
参数简介
通用翻译api的https 地址为,使用方法参考通用翻译api接入文档 。
输入参数
请求方式可使用 get 或 post 方式(content-type 请指定为:application/x-www-form-urlencoded),字符编码统一采用 utf-8 编码格式,单次请求长度控制在 6000 bytes以内(汉字约为输入参数 2000 个)。
将api需要的输入参数封装为一个类(id和密钥换成自己的),代码如下:
/// <summary>
/// 输入产数
/// </summary>
class inputquery
{
/// <summary>
/// app id
/// </summary>
private static string appid = "2021xxxxxxx0626";
/// <summary>
/// 密钥
/// </summary>
private static string password = "cnblongsxtimefiles";
/// <summary>
/// 请求翻译query utf-8编码
/// </summary>
public string content { get; set; }
/// <summary>
/// 翻译源语言 可设置为auto
/// </summary>
public string from { get; set; }
/// <summary>
/// 翻译目标语言 不可设置为auto
/// </summary>
public string to { get; set; }
/// <summary>
/// app id
/// </summary>
public string appid { get; set; }
/// <summary>
/// 随机数 可为字母或数字的字符串
/// </summary>
public string salt { get; set; }
/// <summary>
/// 签名 appid+q+salt+密钥的md5值
/// </summary>
public string sign { get; set; }
/// <summary>
/// 构造函数
/// </summary>
/// <param name="content">翻译内容</param>
/// <param name="fromstr">源语言</param>
/// <param name="tostr">目标语言</param>
public inputquery(string content, string fromstr, string tostr)
{
content = content;
from = fromstr;
to = tostr;
appid = appid;
string randomnum = new random().next().tostring();
string md5sign = getmd5(appid + content + randomnum + password).tolower();
salt = randomnum;
sign = md5sign;
}
/// <summary>
/// 计算字符串的md5值
/// </summary>
static string getmd5(string source)
{
md5cryptoserviceprovider md5 = new md5cryptoserviceprovider();
byte[] data = encoding.utf8.getbytes(source);
byte[] md5data = md5.computehash(data, 0, data.length);
md5.clear();
string deststring = string.empty;
for (int i = 0; i < md5data.length; i++)
{
deststring += convert.tostring(md5data[i], 16).padleft(2, '0');
}
deststring = deststring.padleft(32, '0');
return deststring;
}
}
输出参数
api返回的结果是json格式,需要反序列为对象实例后再进行处理。为了减少第三方库的依赖,反序列化使用.net自带的datacontractjsonserializer。
封装的输出参数类代码如下:
/// <summary>
/// 输出参数
/// </summary>
[datacontract]
class outputresult
{
/// <summary>
/// 错误码 仅当出现错误时显示
/// </summary>
[datamember(name = "error_code")]
public string error_code { set; get; }
/// <summary>
/// 错误信息
/// </summary>
[datamember(name = "error_msg")]
public string error_msg { set; get; }
/// <summary>
/// 源语言 返回用户指定的语言,或者自动检测出的语种(源语言设为 auto 时)
/// </summary>
[datamember(name = "from")]
public string from { set; get; }
/// <summary>
/// 目标语言 返回用户指定的目标语言
/// </summary>
[datamember(name = "to")]
public string to { set; get; }
/// <summary>
/// 翻译结果 返回翻译结果,包括 src 和 dst 字段
/// </summary>
[datamember(name = "trans_result")]
public translatecontent[] trans_result { set; get; }
}
/// <summary>
/// 翻译结果
/// </summary>
[datacontract]
class translatecontent
{
/// <summary>
/// 原文
/// </summary>
[datamember(name = "src")]
public string src { set; get; }
/// <summary>
/// 译文
/// </summary>
[datamember(name = "dst")]
public string dst { set; get; }
}
使用httpclient
调用api需要一个类发送 http 请求以及接收http响应,推荐使用httpclient类(微软不建议使用webclient、httpwebrequest类)。
需要注意的是:httpclient 用于在应用程序的整个生存期内实例化一次并重复使用,也就是说一个应用程序只需要一个httpclient单例即可,代码如下:
public class goodcontroller
{
public static readonly httpclient httpclient;
static goodcontroller()
{
httpclient = new httpclient();
}
}
翻译工具类
将api的调用方法封装成一个translate类,目前只提供一个英翻中的方法,其它翻译需求可以参考 自己扩展。
translate类代码如下(注意添加system.web依赖项):
/// <summary>
/// 百度通用翻译api工具类
/// </summary>
class translate
{
/*
* 代码链接:https://www.cnblogs.com/timefiles/p/baidutranslateapi.html
* 创建时间:2021/06/26
* 主页链接:https://www.cnblogs.com/timefiles/
*
* 常用的语种代码:
* 自动检测 auto
* 中文 zh
* 英语 en
* 日语 jp
*/
/// <summary>
/// 英文翻译为中文
/// </summary>
/// <param name="content">翻译内容</param>
/// <returns></returns>
public static string entozh(string content)
{
string languagefrom = "en", languageto = "zh";
return gettranslateresult(content, languagefrom, languageto);
}
/// <summary>
/// 获取翻译结果
/// </summary>
/// <param name="content">翻译内容</param>
/// <param name="fromstr">源语言</param>
/// <param name="tostr">目标语言</param>
/// <returns></returns>
static string gettranslateresult(string content, string fromstr, string tostr)
{
inputquery input = new inputquery(content, fromstr, tostr);
uri uri = geturi(input);
var task = getoutputresultasync(uri);
//个人认证后api的每秒请求量最高为10,异步意义不大,直接使用同步的方式
outputresult output = task.result;
if (output.error_code == null)
{
return output.trans_result[0].dst;
}
else
{
throw new exception("翻译异常,错误代码:" + output.error_msg);
}
}
/// <summary>
/// 获取请求uri
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
static uri geturi(inputquery input)
{
var query = httputility.parsequerystring(string.empty);
query["q"] = input.content;
query["from"] = input.from;
query["to"] = input.to;
query["appid"] = input.appid;
query["salt"] = input.salt;
query["sign"] = input.sign;
//构造uri,不建议直接拼字符串
uribuilder uribur = new uribuilder("http://api.fanyi.baidu.com/api/trans/vip/translate");
uribur.query = query.tostring();
//uri会自动进行转义
return uribur.uri;
}
/// <summary>
/// 获取api返回的结果
/// </summary>
/// <param name="uri">请求的uri</param>
/// <returns></returns>
static async task<outputresult> getoutputresultasync(uri uri)
{
outputresult result = null;
var response = await goodcontroller.httpclient.getstringasync(uri);
var ser = new datacontractjsonserializer(typeof(outputresult));
using (var ms = new memorystream(encoding.utf8.getbytes(response)))
{
result = (outputresult)ser.readobject(ms);
}
return result;
}
}
个人认证后api的每秒请求量最高为10,此处使用同步方式获取翻译结果,有异步需求的可以自己更改。
应用:翻译halcon的示例描述
static void main(string[] args)
{
xmldocument xd = new xmldocument();
xd.load("index_examples_en_us.xml");
//查找固定名称 节点名要从根节点开始写
xmlnodelist nodelist = xd.documentelement.selectnodes("/examples/example/desc");
for (int i = 0; i < nodelist.count; i++)
{
string desc = nodelist[i].innertext;
//百度翻译有时不太准确,建议保留原文
nodelist[i].innertext = translate.entozh(desc)+"("+ nodelist[i].innertext+")";
console.writeline("翻译结果"+i+":" + nodelist[i].innertext);
console.writeline();
}
streamwriter sw = new streamwriter("index_examples_en_us_翻译.xml", false, new utf8encoding(false));
//为了和原文件保存一致,原文件是使用的是"\n"换行
sw.newline = "\n";
xd.save(sw);
sw.close();
console.writeline("完成");
console.read();
}
注:大概有1000多条需要翻译,需要等待10几分钟。考虑过使用特定字符组合成几个大字符串翻译后再拆分,但特定字符在翻译过程中有丢失导致拆分失败,只能使用这种方式。
将原halcon的xml文件重命名作为备份,将翻译后的文件复制到halcon程序目录下命名为index_examples_en_us.xml,效果如下:
到此这篇关于c#调用百度翻译实现翻译halcon的示例 的文章就介绍到这了,更多相关c# 翻译halcon内容请搜索www.887551.com以前的文章或继续浏览下面的相关文章希望大家以后多多支持www.887551.com!