一. 基础知识

Android 从1.6 版本开始支持Text To Speech ,使用的是Pico 语音合成引擎,但是只支持 English 、 French 、 German 、 Italian 和 Spanish 五大语言,暂时没有对中文提供支持。因此使用 Android 默认的 TTS Engine 是没法朗读中文的。

不过有开源项目 eyes-free (http://code.google.com/p/eyes-free/ , Android 上的 TTS 功能应该也是基于这个开源项目提供的)除了提供 Pico 外,还把支持其他更多语言语音合成的另一个 TTS 引擎 eSpeak 也移植到了 Android 平台,其中就支持中文的语音合成。

因此在安装了 eyes-free 提供的 TTS Service Extended 的 apk 后,就可以在程序中使用 eyes-free 提供的 TTS library ,并把 TTS Engine 设置为不是默认的 Pico ,而是 eSpeak ,就可以实现朗读中文了。不过经过测试,实际的效果还是很差的,只能说勉强可以朗读而已。

二. 实例分析

下面我们就通过一个具体的例子来说明朗读中文的实现过程。

希望实现的效果和之前的类似:

 

图 1 实现效果图

上面一个输入框,点击 “我说” 按钮则朗读上面的文字内容。

首先我们要到 eyes-free 那下载需要的 TTS library 的 jar 包 (http://code.google.com/p/eyes-free/downloads/detail?name=TTS_library_stub_3.0_market.jar&can=2&q=) 。

创建一个 Android 工程,工程名为 NiHaoTTS ,并且把下载的 jar 包放在 assets 文件夹下。右键点击工程,选择 properties , Java Build Path , Libraries , Add JARs, 往工程中添加进 assets 下的 jar 包 ,如下图所示:

图2 添加 TTS library jar 包

其中 main.xml文件很简单,如下所示:

  <?xml version="1.0" encoding="utf-8"?>

 

Java 文件的编写:

Java 文件和之前的类似,只是因为使用了 eyes-free 的 TTS API ,而不是 Android 自带的 TTS API ,因此有些许改变。相关的 API 参考:http://eyes-free.googlecode.com/svn/trunk/documentation/tts/com/google/tts/package-summary.html ,其中我们可以看到基本和 Android 的 API ()类似,其中一个明显的区别是创建 TTS 对象这里使用的是TextToSpeechBeta ,而之前我们使用的是 TextToSpeech


Java完整的代码为:

[java] view plain copy print ?
  1. public class NiHaoTTS extends Activity implements OnInitListener{  
  2.     /** Called when the activity is first created. */  
  3.     private Button mBtn;  
  4.     private EditText mText;  
  5.     //使用com.google.tts包中的TextToSpeechBeta  
  6.     private TextToSpeechBeta mTTS;  
  7.       
  8.     private static final String TAG = "TTS Demo";    
  9.     private static final int REQ_TTS_STATUS_CHECK = 0;    
  10.       
  11.     @Override  
  12.     public void onCreate(Bundle savedInstanceState) {  
  13.         super.onCreate(savedInstanceState);  
  14.         setContentView(R.layout.main);  
  15.           
  16.         //检查TTS数据是否已经安装并且可用   
  17.         Intent checkIntent = new Intent();  
  18.         checkIntent.setAction(TextToSpeechBeta.Engine.ACTION_CHECK_TTS_DATA);  
  19.         startActivityForResult(checkIntent, REQ_TTS_STATUS_CHECK);  
  20.           
  21.         mText = (EditText)findViewById(R.id.ttstext);  
  22.         mBtn = (Button) findViewById(R.id.ttsbtn);  
  23.         mBtn.setOnClickListener(new OnClickListener() {  
  24.               
  25.             @Override  
  26.             public void onClick(View v) {  
  27.                 // TODO Auto-generated method stub  
  28.                 String ttsText = mText.getText().toString();  
  29.                 if(ttsText != "")  
  30.                 {  
  31.                     //读取文本框中的中文  
  32.                     mTTS.speak(ttsText, TextToSpeechBeta.QUEUE_ADD, null);  
  33.                 }  
  34.             }  
  35.         });  
  36.     }  
  37.   //实现TTS初始化接口     
  38.     @Override  
  39.     public void onInit(int status, int version) {  
  40.         // TODO Auto-generated method stub  
  41.         Log.v(TAG, "version = " + String.valueOf(version));  
  42.         //判断TTS初始化的返回版本号,如果为-1,表示没有安装对应的TTS数据  
  43.         if(version == -1)  
  44.         {  
  45.             //提示安装所需的TTS数据   
  46.             alertInstallEyesFreeTTSData();  
  47.         }  
  48.         else  
  49.         {  
  50.             //TTS Engine初始化完成   
  51.             if(status == TextToSpeechBeta.SUCCESS)  
  52.             {  
  53.                 Log.v(TAG, "success to init tts");  
  54.                 //设置TTS引擎,com.google.tts即eSpeak支持的语言包含中文,使用Android系统默认的pico可以设置为com.svox.pico  
  55.                 mTTS.setEngineByPackageNameExtended("com.google.tts");  
  56.                 int result = mTTS.setLanguage(Locale.CHINA);  
  57.                 //设置发音语言   
  58.                 if(result == TextToSpeechBeta.LANG_MISSING_DATA || result == TextToSpeechBeta.LANG_NOT_SUPPORTED)  
  59.                 //判断语言是否可用   
  60.                 {  
  61.                     Log.v(TAG, "Language is not available");  
  62.                     mBtn.setEnabled(false);  
  63.                 }  
  64.                 else  
  65.                 {  
  66.                     mTTS.speak("你好,朋友!", TextToSpeechBeta.QUEUE_ADD, null);  
  67.                     mBtn.setEnabled(true);  
  68.                 }  
  69.             }  
  70.             else  
  71.             {  
  72.                 Log.v(TAG, "failed to init tts");  
  73.             }  
  74.         }  
  75.     }  
  76.     protected  void onActivityResult(int requestCode, int resultCode, Intent data) {  
  77.         if(requestCode == REQ_TTS_STATUS_CHECK)  
  78.         {  
  79.             switch (resultCode) {  
  80.             case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_PASS:  
  81.                 //这个返回结果表明TTS Engine可以用  
  82.             {  
  83.                 //使用的是TextToSpeechBeta  
  84.                 mTTS = new TextToSpeechBeta(thisthis);  
  85.                 Log.v(TAG, "TTS Engine is installed!");    
  86.                   
  87.             }  
  88.                   
  89.                 break;  
  90.             case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_BAD_DATA:  
  91.                 //需要的语音数据已损坏   
  92.             case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_MISSING_DATA:  
  93.                 //缺少需要语言的语音数据   
  94.             case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_MISSING_VOLUME:  
  95.                 //缺少需要语言的发音数据   
  96.             {  
  97.                 //这三种情况都表明数据有错,重新下载安装需要的数据  
  98.                 Log.v(TAG, "Need language stuff:"+resultCode);  
  99.                 Intent dataIntent = new Intent();  
  100.                 dataIntent.setAction(TextToSpeechBeta.Engine.ACTION_INSTALL_TTS_DATA);  
  101.                 startActivity(dataIntent);  
  102.             }  
  103.                 break;  
  104.             case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_FAIL:  
  105.                 //检查失败   
  106.             default:  
  107.                 Log.v(TAG, "Got a failure. TTS apparently not available");  
  108.                 break;  
  109.             }  
  110.         }  
  111.         else  
  112.         {  
  113.             //其他Intent返回的结果   
  114.         }  
  115.     }  
  116.     //弹出对话框提示安装所需的TTS数据   
  117.     private void alertInstallEyesFreeTTSData()  
  118.     {  
  119.         Builder alertInstall = new AlertDialog.Builder(this)  
  120.             .setTitle("缺少需要的语音包")  
  121.             .setMessage("下载安装缺少的语音包")  
  122.             .setPositiveButton("确定"new DialogInterface.OnClickListener() {  
  123.                   
  124.                 @Override  
  125.                 public void onClick(DialogInterface dialog, int which) {  
  126.                     // TODO Auto-generated method stub  
  127.                     //下载eyes-free的语音数据包  
  128.                     String ttsDataUrl = "http://eyes-free.googlecode.com/files/tts_3.1_market.apk";  
  129.                     Uri ttsDataUri = Uri.parse(ttsDataUrl);  
  130.                     Intent ttsIntent = new Intent(Intent.ACTION_VIEW, ttsDataUri);  
  131.                     startActivity(ttsIntent);  
  132.                 }  
  133.             })  
  134.             .setNegativeButton("取消"new DialogInterface.OnClickListener() {  
  135.                   
  136.                 @Override  
  137.                 public void onClick(DialogInterface dialog, int which) {  
  138.                     // TODO Auto-generated method stub  
  139.                     finish();  
  140.                 }  
  141.             });  
  142.             alertInstall.create().show();  
  143.           
  144.     }  
  145.       
  146.     @Override  
  147.     protected void onDestroy() {  
  148.         // TODO Auto-generated method stub  
  149.         super.onDestroy();  
  150.         if(mTTS!=null){  
  151.             mTTS.shutdown();  
  152.         }  
  153.     }  
  154.     @Override  
  155.     protected void onPause() {  
  156.         // TODO Auto-generated method stub  
  157.         super.onPause();  
  158.         if(mTTS != null)  
  159.         {  
  160.             mTTS.stop();  
  161.         }  
  162.     }  
  163.       
  164. }  
public class NiHaoTTS extends Activity implements OnInitListener{ /** Called when the activity is first created. */ private Button mBtn; private EditText mText; //使用com.google.tts包中的TextToSpeechBeta private TextToSpeechBeta mTTS; private static final String TAG = "TTS Demo"; private static final int REQ_TTS_STATUS_CHECK = 0; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); //检查TTS数据是否已经安装并且可用 Intent checkIntent = new Intent(); checkIntent.setAction(TextToSpeechBeta.Engine.ACTION_CHECK_TTS_DATA); startActivityForResult(checkIntent, REQ_TTS_STATUS_CHECK); mText = (EditText)findViewById(R.id.ttstext); mBtn = (Button) findViewById(R.id.ttsbtn); mBtn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub String ttsText = mText.getText().toString(); if(ttsText != "") { //读取文本框中的中文 mTTS.speak(ttsText, TextToSpeechBeta.QUEUE_ADD, null); } } }); } //实现TTS初始化接口 @Override public void onInit(int status, int version) { // TODO Auto-generated method stub Log.v(TAG, "version = " + String.valueOf(version)); //判断TTS初始化的返回版本号,如果为-1,表示没有安装对应的TTS数据 if(version == -1) { //提示安装所需的TTS数据 alertInstallEyesFreeTTSData(); } else { //TTS Engine初始化完成 if(status == TextToSpeechBeta.SUCCESS) { Log.v(TAG, "success to init tts"); //设置TTS引擎,com.google.tts即eSpeak支持的语言包含中文,使用Android系统默认的pico可以设置为com.svox.pico mTTS.setEngineByPackageNameExtended("com.google.tts"); int result = mTTS.setLanguage(Locale.CHINA); //设置发音语言 if(result == TextToSpeechBeta.LANG_MISSING_DATA || result == TextToSpeechBeta.LANG_NOT_SUPPORTED) //判断语言是否可用 { Log.v(TAG, "Language is not available"); mBtn.setEnabled(false); } else { mTTS.speak("你好,朋友!", TextToSpeechBeta.QUEUE_ADD, null); mBtn.setEnabled(true); } } else { Log.v(TAG, "failed to init tts"); } } } protected void onActivityResult(int requestCode, int resultCode, Intent data) { if(requestCode == REQ_TTS_STATUS_CHECK) { switch (resultCode) { case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_PASS: //这个返回结果表明TTS Engine可以用 { //使用的是TextToSpeechBeta mTTS = new TextToSpeechBeta(this, this); Log.v(TAG, "TTS Engine is installed!"); } break; case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_BAD_DATA: //需要的语音数据已损坏 case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_MISSING_DATA: //缺少需要语言的语音数据 case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_MISSING_VOLUME: //缺少需要语言的发音数据 { //这三种情况都表明数据有错,重新下载安装需要的数据 Log.v(TAG, "Need language stuff:"+resultCode); Intent dataIntent = new Intent(); dataIntent.setAction(TextToSpeechBeta.Engine.ACTION_INSTALL_TTS_DATA); startActivity(dataIntent); } break; case TextToSpeechBeta.Engine.CHECK_VOICE_DATA_FAIL: //检查失败 default: Log.v(TAG, "Got a failure. TTS apparently not available"); break; } } else { //其他Intent返回的结果 } } //弹出对话框提示安装所需的TTS数据 private void alertInstallEyesFreeTTSData() { Builder alertInstall = new AlertDialog.Builder(this) .setTitle("缺少需要的语音包") .setMessage("下载安装缺少的语音包") .setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // TODO Auto-generated method stub //下载eyes-free的语音数据包 String ttsDataUrl = "http://eyes-free.googlecode.com/files/tts_3.1_market.apk"; Uri ttsDataUri = Uri.parse(ttsDataUrl); Intent ttsIntent = new Intent(Intent.ACTION_VIEW, ttsDataUri); startActivity(ttsIntent); } }) .setNegativeButton("取消", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // TODO Auto-generated method stub finish(); } }); alertInstall.create().show(); } @Override protected void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); if(mTTS!=null){ mTTS.shutdown(); } } @Override protected void onPause() { // TODO Auto-generated method stub super.onPause(); if(mTTS != null) { mTTS.stop(); } } }

 

基本和我们之前学过的类似,其中不同的地方有:


1 TTS 的初始化回调函数多了一个版本号参数 version,

publicvoidonInit(intstatus,intversion),

如果返回的版本号version等于 -1,则表明还没有安装 eyes-free提供的 TTS Service Extended,提示进行下载和安装。

//提示安装所需的TTS 数据

alertInstallEyesFreeTTSData();


主要就是弹出对话框进行提示,然后链接到eyes-free 的网站进行下载对应的apk

//弹出对话框提示安装所需的TTS 数据

privatevoidalertInstallEyesFreeTTSData()

{

Builder alertInstall =new AlertDialog.Builder(this )

.setTitle(" 缺少需要的语音包 " )

.setMessage(" 下载安装缺少的语音包" )

.setPositiveButton(" 确定 " , newDialogInterface.OnClickListener() {

@Override

publicvoidonClick(DialogInterface dialog, intwhich) {

//TODO Auto-generated method stub

//下载 eyes-free的语音数据包

String ttsDataUrl ="http://eyes-free.googlecode.com/files/tts_3.1_market.apk";

Uri ttsDataUri = Uri.parse (ttsDataUrl);

Intent ttsIntent =new Intent(Intent.ACTION_VIEW , ttsDataUri);

startActivity(ttsIntent);

}

})

.setNegativeButton(" 取消 " , newDialogInterface.OnClickListener() {

@Override

publicvoidonClick(DialogInterface dialog, intwhich) {

//TODO Auto-generated method stub

finish();

}

});

alertInstall.create().show();

}


下载完进行安装,然后重新启动程序就可以运行。

2)如果已经安装了,则可以进设置TTS引擎和需要合成的语言。

//设置 TTS 引擎, com.google.tts eSpeak 支持的语言包含中文,使用Android系统默认的pico可以设置为com.svox.pico

mTTS.setEngineByPackageNameExtended( "com.google.tts" );

intresult = mTTS.setLanguage(Locale. CHINA );

//设置发音语言


其中如果设置为com.google.tts 则表示使用 eSpeak 引擎,其包括对中文语音合成的支持。如果设置为 com.svox.pico则表示使用 Pico引擎,实际效果就是使用 Android自带的 Pico的效果,不过只支持 5中语言,不支持中文。


除了这两块,其他和我们之前的程序基本一样。

然后就可以运行程序。因为模拟器开始时没有安装需要的eyes-free TTS Service Extended apk,因此会弹出提示对话框,点击确定开始下载,下载完进行安装,如下图所示。

图3 安装eyes-free的TTS

安装完会在应用程序中看到对应的图标,并且在TTS 的设置中也会增加 eSpeak TTS一项,如下图所示:

图4 安装了eyes-free的TTS

安装完后就可以再次打开NiHaoTTS 程序,这时会有个选择询问使用哪个 TTS,选择第二项,然后就可以在输入框中输入中文来朗读了。

图5 选择TTS

不过经过简单测试就知道了,中文朗读的效果基本没法实际使用的。

 

文章对应的完整代码例子可以在这里下载:

http://download.csdn.net/source/3237373

更多相关文章

  1. Android数据绑定Data Binding初体验
  2. Android Studio部署apk到手机后手机显示中文乱码解决方法
  3. Android 绑定数据到界面控件
  4. Android使用AIDL跨进程数据共享
  5. Android 传感器的 数据流和框架
  6. Android 数据操作之SQLiteDatabase
  7. 获取PCM音频数据的声音分贝值
  8. android官网 100%的中文翻译版
  9. Android 从properties配置文件读取数据

随机推荐

  1. Android的应用程序的异常处理2
  2. Android(安卓)OpenGL ES2.0 and GLSL 一
  3. Ubuntu 交叉编译 OpenSSL,并在 Android(安
  4. android浏览器开发小技巧集锦
  5. Android(安卓)studio开发环境搭建教程与
  6. Android(安卓)4.0 源码 下 载
  7. 文章标题android自制多媒体视频播放器和
  8. 一句话锁定MySQL数据占用元凶
  9. 适合Material Dsign的新抽屉---Navigatio
  10. 使用Android手机APP查看ROS中RGB摄像头数