当前位置:   article > 正文

百度语音合成(TTS)离在线融合,包括-102问题的完美解决!

百度语音合成(TTS)离在线融合,包括-102问题的完美解决!

最近项目要用到语音合成(TTS)的功能,目前国内只有两家:科大讯飞:科大讯飞开发者平台  和 百度:百度语音 。科大讯飞支持完全离线,但是需要付费(成本高),又因为百度虽然目前没有开放纯离线模式下的语音合成SDK,但是开放了离在线融合语音合成SDK,会自动判断当前网络环境,自动匹配使用离线还是在线合成引擎。简言之,就是根据网络的不同状态,实现离线条件下的语音合成。

首先我们还是来确认一下使用百度TTS的流程:

1下载官方sdk。(这个不多说)

2申请apiID,这个是重点,说白了102的错误归根结底出现在这里。我们来看图



这个是百度云平台的语音合成管理https://cloud.baidu.com

首先语音合成必选,这是你的目的;其次文字识别得选(离线需要这个权限),没有这个受不了权就会出现-102授权失败的问题;然后包名,平台都不用说。

然后就是导包,把官方下载的demo中的data改名assets删掉里面的temp-listenes,放到main平级的目录。

最后就是敲代码了:下面我把这个语音合成给封装了一下:

  1. /**
  2. * Created by MaTengBin on 2017/7/13.
  3. */
  4. public class SpeechUtil implements SpeechSynthesizerListener {
  5. private SpeechSynthesizer mSpeechSynthesizer;
  6. private String mSampleDirPath;
  7. private static final String SAMPLE_DIR_NAME = "baiduTTS";
  8. private static final String SPEECH_FEMALE_MODEL_NAME = "bd_etts_speech_female.dat";
  9. private static final String SPEECH_MALE_MODEL_NAME = "bd_etts_speech_male.dat";
  10. private static final String TEXT_MODEL_NAME = "bd_etts_text.dat";
  11. private static final String LICENSE_FILE_NAME = "temp_license";
  12. private static final String ENGLISH_SPEECH_FEMALE_MODEL_NAME = "bd_etts_speech_female_en.dat";
  13. private static final String ENGLISH_SPEECH_MALE_MODEL_NAME = "bd_etts_speech_male_en.dat";
  14. private static final String ENGLISH_TEXT_MODEL_NAME = "bd_etts_text_en.dat";
  15. private Context context;
  16. public SpeechUtil(Context context){
  17. this.context = context;
  18. initialEnv();
  19. initialTts();
  20. }
  21. private void initialEnv() {
  22. if (mSampleDirPath == null) {
  23. String sdcardPath = Environment.getExternalStorageDirectory().toString();
  24. mSampleDirPath = sdcardPath + "/" + SAMPLE_DIR_NAME;
  25. }
  26. makeDir(mSampleDirPath);
  27. copyFromAssetsToSdcard(false, SPEECH_FEMALE_MODEL_NAME, mSampleDirPath + "/" + SPEECH_FEMALE_MODEL_NAME);
  28. copyFromAssetsToSdcard(false, SPEECH_MALE_MODEL_NAME, mSampleDirPath + "/" + SPEECH_MALE_MODEL_NAME);
  29. copyFromAssetsToSdcard(false, TEXT_MODEL_NAME, mSampleDirPath + "/" + TEXT_MODEL_NAME);
  30. copyFromAssetsToSdcard(false, LICENSE_FILE_NAME, mSampleDirPath + "/" + LICENSE_FILE_NAME);
  31. copyFromAssetsToSdcard(false, "english/" + ENGLISH_SPEECH_FEMALE_MODEL_NAME, mSampleDirPath + "/"
  32. + ENGLISH_SPEECH_FEMALE_MODEL_NAME);
  33. copyFromAssetsToSdcard(false, "english/" + ENGLISH_SPEECH_MALE_MODEL_NAME, mSampleDirPath + "/"
  34. + ENGLISH_SPEECH_MALE_MODEL_NAME);
  35. copyFromAssetsToSdcard(false, "english/" + ENGLISH_TEXT_MODEL_NAME, mSampleDirPath + "/"
  36. + ENGLISH_TEXT_MODEL_NAME);
  37. }
  38. private void makeDir(String dirPath) {
  39. File file = new File(dirPath);
  40. if (!file.exists()) {
  41. file.mkdirs();
  42. }
  43. }
  44. /**
  45. * 将sample工程需要的资源文件拷贝到SD卡中使用(授权文件为临时授权文件,请注册正式授权)
  46. *
  47. * @param isCover 是否覆盖已存在的目标文件
  48. * @param source
  49. * @param dest
  50. */
  51. private void copyFromAssetsToSdcard(boolean isCover, String source, String dest) {
  52. File file = new File(dest);
  53. if (isCover || (!isCover && !file.exists())) {
  54. InputStream is = null;
  55. FileOutputStream fos = null;
  56. try {
  57. is = context.getResources().getAssets().open(source);
  58. String path = dest;
  59. fos = new FileOutputStream(path);
  60. byte[] buffer = new byte[1024];
  61. int size = 0;
  62. while ((size = is.read(buffer, 0, 1024)) >= 0) {
  63. fos.write(buffer, 0, size);
  64. }
  65. } catch (FileNotFoundException e) {
  66. e.printStackTrace();
  67. } catch (IOException e) {
  68. e.printStackTrace();
  69. } finally {
  70. if (fos != null) {
  71. try {
  72. fos.close();
  73. } catch (IOException e) {
  74. e.printStackTrace();
  75. }
  76. }
  77. try {
  78. if (is != null) {
  79. is.close();
  80. }
  81. } catch (IOException e) {
  82. e.printStackTrace();
  83. }
  84. }
  85. }
  86. }
  87. private void initialTts() {
  88. this.mSpeechSynthesizer = SpeechSynthesizer.getInstance();
  89. this.mSpeechSynthesizer.setContext(context);
  90. this.mSpeechSynthesizer.setSpeechSynthesizerListener(this);
  91. // 文本模型文件路径 (离线引擎使用)
  92. this.mSpeechSynthesizer.setParam(SpeechSynthesizer.PARAM_TTS_TEXT_MODEL_FILE, mSampleDirPath + "/"
  93. + TEXT_MODEL_NAME);
  94. // 声学模型文件路径 (离线引擎使用)
  95. this.mSpeechSynthesizer.setParam(SpeechSynthesizer.PARAM_TTS_SPEECH_MODEL_FILE, mSampleDirPath + "/"
  96. + SPEECH_FEMALE_MODEL_NAME);
  97. // 本地授权文件路径,如未设置将使用默认路径.设置临时授权文件路径,LICENCE_FILE_NAME请替换成临时授权文件的实际路径,仅在使用临时license文件时需要进行设置,如果在[应用管理]中开通了正式离线授权,不需要设置该参数,建议将该行代码删除(离线引擎)
  98. // 如果合成结果出现临时授权文件将要到期的提示,说明使用了临时授权文件,请删除临时授权即可。
  99. // this.mSpeechSynthesizer.setParam(SpeechSynthesizer.PARAM_TTS_LICENCE_FILE, mSampleDirPath + "/"
  100. // + LICENSE_FILE_NAME);
  101. // 请替换为语音开发者平台上注册应用得到的App ID (离线授权)
  102. this.mSpeechSynthesizer.setAppId("9913010"/*这里只是为了让Demo运行使用的APPID,请替换成自己的id。*/);
  103. // 请替换为语音开发者平台注册应用得到的apikey和secretkey (在线授权)
  104. this.mSpeechSynthesizer.setApiKey("9iIhf2hjqH975Lhqte5FlmnU",
  105. "d9db97caef194d5eb5e79900da9613c4"/*这里只是为了让Demo正常运行使用APIKey,请替换成自己的APIKey*/);
  106. // 发音人(在线引擎),可用参数为0,1,2,3。。。(服务器端会动态增加,各值含义参考文档,以文档说明为准。0--普通女声,1--普通男声,2--特别男声,3--情感男声。。。)
  107. this.mSpeechSynthesizer.setParam(SpeechSynthesizer.PARAM_SPEAKER, "0");
  108. // 设置Mix模式的合成策略
  109. this.mSpeechSynthesizer.setParam(SpeechSynthesizer.PARAM_MIX_MODE, SpeechSynthesizer.MIX_MODE_DEFAULT);
  110. // 初始化tts
  111. mSpeechSynthesizer.initTts(TtsMode.MIX);
  112. // 加载离线英文资源(提供离线英文合成功能)
  113. mSpeechSynthesizer.loadEnglishModel(mSampleDirPath + "/" + ENGLISH_TEXT_MODEL_NAME, mSampleDirPath
  114. + "/" + ENGLISH_SPEECH_FEMALE_MODEL_NAME);
  115. }
  116. String ago = "";
  117. public void speak(String s) {
  118. String text = s.toString();
  119. //需要合成的文本text的长度不能超过1024个GBK字节。
  120. if (!TextUtils.isEmpty(text) && !ago.equals(text)) {
  121. mSpeechSynthesizer.speak(text);
  122. ago = text;
  123. }
  124. }
  125. @Override
  126. public void onSynthesizeStart(String s) {
  127. }
  128. @Override
  129. public void onSynthesizeDataArrived(String s, byte[] bytes, int i) {
  130. }
  131. @Override
  132. public void onSynthesizeFinish(String s) {
  133. }
  134. @Override
  135. public void onSpeechStart(String s) {
  136. }
  137. @Override
  138. public void onSpeechProgressChanged(String s, int i) {
  139. }
  140. @Override
  141. public void onSpeechFinish(String s) {
  142. mSpeechSynthesizer.stop();
  143. }
  144. @Override
  145. public void onError(String s, SpeechError speechError) {
  146. }
  147. }

调用就更简单了:

  1. SpeechUtil speechUtil = new SpeechUtil(context);
  2. speechUtil.speak(s);

这个就是全过程了,如果这篇文章解决了你的问题,请点赞,谢谢!

源码下载地址:http://download.csdn.net/detail/m0_38082472/9905307

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/124133
推荐阅读
相关标签
  

闽ICP备14008679号