zoukankan      html  css  js  c++  java
  • 安卓TTS语音合成经验分享(科大讯飞语音+)集成

    应用场景:足浴软件,技师钟房安排调派和队列排序查看,语音播报提醒。老程序是使用双屏显卡,windows系统PC上运行一个无人值守桌面程序。如今安卓机顶盒(WIFI)和MINI电视棒通过HDMI接口和支持大屏幕高清液晶显示器(电视)组合,让这一应用场景的成本大幅下降,实用性更好。

    通过一番调研,发现国内上市公司科大讯飞的语音+SDK包口碑好,效果好,集成也方便。语音+支持离线应用,这是最主要的。集成过程走起:

    1、下载DEMO程序。DEMO程序就包含了全部需要的资源。

    2、SDK集成关键代码:

    (1)SpeechSynthesizer 是 jar包里封装的语音合成对象她是我们的女主角。

    // 语音合成对象
    private SpeechSynthesizer mTts;
    public static String SPEAKER = "speaker";
    

    从demo照抄实例化mTts的代码。调用方式应该是jar里封装好的方法外包调用,暂时不关注,先关注一下怎样用起来。

    private void checkVoiceServiceIsInstalled() {
    		// 检测语音引擎是否可用,如果没有安装自动安装
    		if (!checkSpeechServiceInstall()) {
    			String assetsApk = "SpeechService.apk";
    			if (processInstall(this, assetsApk))
    				SpeechUtility.getUtility(this).setAppid("53c776aa");
    			return;
    		}
    		// 引擎初始化
    		SpeechUtility.getUtility(this).setAppid("53c776aa");
    		mTts = new SpeechSynthesizer(this, mTtsInitListener);
    	}
    

     (2)检查语音加引擎是否安装 

    /**
    	 * 执行本地安装 语音+
    	 * 
    	 * @param context
    	 * @param assetsApk
    	 * @return
    	 */
    	private boolean processInstall(Context context, String assetsApk) {
    		try {
    			Thread.sleep(30000);
    		} catch (InterruptedException e) {
    			e.printStackTrace();
    		}
    		// 本地安装方式
    		if (!ApkInstaller.installFromAssets(context, assetsApk)) {
    			Toast.makeText(activity_main.this, "安装语音引擎失败!", Toast.LENGTH_SHORT)
    					.show();
    			return false;
    		}
    		return true;
    	}
    
    	/**
    	 * 检测科大讯飞语音+引擎是否安装
    	 * 
    	 * @return
    	 */
    	private boolean checkSpeechServiceInstall() {
    		String packageName = "com.iflytek.speechcloud";
    		List<PackageInfo> packages = getPackageManager()
    				.getInstalledPackages(0);
    		for (int i = 0; i < packages.size(); i++) {
    			PackageInfo packageInfo = packages.get(i);
    			if (packageInfo.packageName.equals(packageName)) {
    				return true;
    			} else {
    				continue;
    			}
    		}
    		return false;
    	}
    
    	/**
    	 * 语音合成的参数设置
    	 * 
    	 * @param param
    	 * @return
    	 */
    	private void setParam() {
    		mTts.setParameter(SpeechConstant.ENGINE_TYPE,
    				sp.getString("engine_preference", "local"));
    
    		if (sp.getString("engine_preference", "local")
    				.equalsIgnoreCase("local")) {
    			mTts.setParameter(SpeechSynthesizer.VOICE_NAME,
    					sp.getString("role_cn_preference", "xiaoyan"));
    		} else {
    			mTts.setParameter(SpeechSynthesizer.VOICE_NAME,
    					sp.getString("role_cn_preference", "xiaoyan"));
    		}
    		mTts.setParameter(SpeechSynthesizer.SPEED,
    				sp.getString("speed_preference", "50"));
    
    		mTts.setParameter(SpeechSynthesizer.PITCH,
    				sp.getString("pitch_preference", "50"));
    
    		mTts.setParameter(SpeechSynthesizer.VOLUME,
    				sp.getString("volume_preference", "50"));
    	}
    

      (3)初期化监听,我在ErrorCode.SUCCESS初始化成功后显示语音引擎可用的状态,否则红叉显示不可以状态。合成回调监听,这里为了避免异步方式调用,听不清楚,使用了局部变量记录当前的语音播报状态isPlaySound。引擎支持异步调用,如果不控制会出现,一句话没说完,其他调用又在发起请求的情况,使用了队列来控制播报的顺序。

    /**
    	 * 初期化监听。
    	 */
    	private InitListener mTtsInitListener = new InitListener() {
    
    		@Override
    		public void onInit(ISpeechModule arg0, int code) {
    			if (code == ErrorCode.SUCCESS) {
    				btn_voiceDemo
    						.setBackgroundResource(R.drawable.voice_enable_style);
    			} else {
    				btn_voiceDemo
    						.setBackgroundResource(R.drawable.voice_disable_style);
    			}
    		}
    	};
    
    	/**
    	 * 合成回调监听。
    	 */
    	private SynthesizerListener mTtsListener = new SynthesizerListener.Stub() {
    		@Override
    		public void onBufferProgress(int progress) throws RemoteException {
    
    		}
    
    		@Override
    		public void onCompleted(int code) throws RemoteException {
    			activity_main.this.runOnUiThread(new Runnable() {
    				@Override
    				public void run() {
    					int pos = voiceQueueList.indexOf(getCurrentVoiceEntity());
    					if (pos != -1) {
    						currentVoiceEntity.setVoicePlayNum(currentVoiceEntity
    								.getVoicePlayNum() + 1);
    						voiceQueueList.remove(pos);
    						if (currentVoiceEntity.getVoicePlayNum() > Integer
    								.parseInt(sp
    										.getString("voice_play_timers", "0"))) {
    							currentVoiceEntity.setIsVoicePlayed(true);
    							new ExecVoiceStatusChangeBiz().execute(String
    									.valueOf(getCurrentVoiceEntity().getKeyId()));
    						}
    						voiceQueueList.add(currentVoiceEntity);
    					}
    					loadVoiceTaskList();
    				}
    			});
    			isPlaySound = false;
    		}
    
    		@Override
    		public void onSpeakBegin() throws RemoteException {
    			isPlaySound = true;
    		}
    
    		@Override
    		public void onSpeakPaused() throws RemoteException {
    
    		}
    
    		@Override
    		public void onSpeakProgress(int progress) throws RemoteException {
    
    		}
    
    		@Override
    		public void onSpeakResumed() throws RemoteException {
    
    		}
    	};
    
    	/**
    	 * 窗体销毁时释放语音服务资源
    	 */
    	@Override
    	protected void onDestroy() {
    		super.onDestroy();
    		mTts.stopSpeaking(mTtsListener);
    		mTts.destory();
    	}
    

     3、完成。截图纪念一下:

     

      

       

  • 相关阅读:
    Machine Learning/Introducing Logistic Function
    学习Machine Leaning In Action(四):逻辑回归
    极致和厚道为小米新国货的核心实质(极致不是指最好的,而是要超乎预期)
    做事要提前留有余地,否则到时候就来不及了(超市买菜,找工作,交朋友,发脾气都是如此)
    UpdateWindow API函数的作用很明显
    近几年前端技术盘点以及 2016 年技术发展方向
    使用SetLocaleInfo设置时间后必须调用广播WM_SETTINGCHANGE,通知其他程序格式已经更改
    设计模式——(Abstract Factory)抽象工厂
    从优化到再优化,最长公共子串
    与数据库打交道的Adapter----SimpleCursorAdapter
  • 原文地址:https://www.cnblogs.com/datacool/p/datacool_android_tts.html
Copyright © 2011-2022 走看看