在配置好Android Studio 2.3.3后,依照结合网上例子,动手创建讯飞语言听写app,最终手机上运行成功。

主要参考两篇博文(zqHero/XunFeiVoiceDEmo ,Android Studio 2.3.3 接入讯飞语音接口),都非常好,基本步骤参考第一篇,代码参考第二篇。

  1. 创建讯飞应用:在讯飞开发平台注册账号,在网站上创建应用,得到一个APPID,这个ID用来绑定讯飞SDK和自己制作的APP

  2. 创建SDK:选择所需服务(如语言听写)、平台(Android)、和应用(第一步创建的应用),然后下载该SDK。SDK主要有三个文件夹,/assets,/libs,/sample.

  3. 创建android Studio项目:新建项目,修改项目名(如VoiceDemo)后其余都为默认,创建一个helloworld项目,文件浏览方式默认为Android,改为Project方式。

  4. 复制.jar文件:将SDK中/libs文件夹中的.jar文件都复制到android项目VoiceDemo/app/libs/文件夹下。可能由于生成SDK时只选择语言听写,只有两个.jar文件(分别为Msc.jar,Sunflower.jar)

  5. 复制.so文件:先在android项目VoiceDemo/app/src/main/下新建文件夹/jniLibs,将SDK中/libs下的包含.so文件夹全部复制到VoiceDemo/app/src/main/jniLibs/文件夹下。总共七个文件夹分别是/arm64-v8a,/armeabi,/armeabi-v7a,/mips,/mips64,/x86,/86_64,每个文件夹下都只有一个名为libmsc.so的文件

  6. 复制/assets文件夹:将SDK中/assets文件夹复制到VoiceDemo/app/src/main/

  7. 刷新.gradle文件:选中VoiceDemo/app下的build.gradle,右键选择Synchronize 'build.gradle',依次刷新其它两个.gradle文件。工具栏中也有Synch Project with Gradle Files*可能达到同样效果。在这一步中.gradle中没有出现文章二中的代码,我也没有手动添加,但好像影响不大。初次接触Android编程,不懂原理,希望热心人解答。

  8. 编辑权限:在*VoiceDemo/app/src/main/AndroidManifest.xml文件中依照第二篇文章代码添加相应权限;

  9. 编辑布局:在*VoiceDemo/app/src/main/res/layout/activity_main.xml文件中依照第二篇文章添加按钮和文本框

  10. 编辑功能:在*VoiceDemo/app/src/main/java/example.org.voicedemo/MainActivity.java文件中依照第二篇文章添加给按钮添加功能,将在讯飞平台创建得到的APPID添加进去。在这一步时import com.iflytek包全部显示有红色下划线,提示找不到com.iflytek。我将项目关闭后重新打开红色下划线就不存在了。这个bug出现的很奇怪,解决的也很奇怪。希望有热心人解答。

  11. Run app:选择在模拟其中运行,能够显示主页面布局,点击按钮录音时提示权限不对,但却没有同意权限的选择弹出。选择在手机中安装运行,报错-26.

  12. 打包生成APK:在Build工具选项下选择Build APK将生成的APK拷贝到手机中安装,同意相应权限后,语音听写功能正常运行。如第二篇文章。

项目文件的代码:

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
   package="example.org.voicedemo">

   <application
       android:allowBackup="true"
       android:icon="@mipmap/ic_launcher"
       android:label="@string/app_name"
       android:roundIcon="@mipmap/ic_launcher_round"
       android:supportsRtl="true"
       android:theme="@style/AppTheme">
       <activity android:name=".MainActivity">
           <intent-filter>
               <action android:name="android.intent.action.MAIN" />

               <category android:name="android.intent.category.LAUNCHER" />
           </intent-filter>
       </activity>
   </application>

   <uses-permission android:name="android.permission.INTERNET" />
   <uses-permission android:name="android.permission.RECORD_AUDIO" />
   <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
   <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
   <!--读取网络信息状态 -->
   <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
   <!--获取当前wifi状态 -->
   <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
   <!--允许程序改变网络连接状态 -->
   <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE"/>
   <!--读取手机信息权限 -->
   <uses-permission android:name="android.permission.READ_PHONE_STATE"/>
   <uses-permission android:name="android.permission.WRITE_SETTINGS" />


</manifest>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:app="http://schemas.android.com/apk/res-auto"
   xmlns:tools="http://schemas.android.com/tools"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   tools:context="example.org.voicedemo.MainActivity">

   <TextView
       android:id="@+id/textView"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:text="Hello World!"
       tools:layout_constraintTop_creator="1"
       tools:layout_constraintRight_creator="1"
       tools:layout_constraintBottom_creator="1"
       app:layout_constraintBottom_toBottomOf="parent"
       app:layout_constraintRight_toRightOf="parent"
       tools:layout_constraintLeft_creator="1"
       app:layout_constraintLeft_toLeftOf="parent"
       app:layout_constraintTop_toTopOf="parent" />

   <Button
       android:id="@+id/button"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:text="Button"
       tools:layout_constraintTop_creator="1"
       tools:layout_constraintRight_creator="1"
       app:layout_constraintRight_toRightOf="parent"
       android:layout_marginTop="20dp"
       app:layout_constraintTop_toBottomOf="@+id/textView"
       tools:layout_constraintLeft_creator="1"
       app:layout_constraintLeft_toLeftOf="parent" />



</android.support.constraint.ConstraintLayout>

MainActivity.java

package example.org.voicedemo;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

import android.os.Environment;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import com.iflytek.cloud.RecognizerResult;
import com.iflytek.cloud.SpeechConstant;
import com.iflytek.cloud.SpeechError;
import com.iflytek.cloud.SpeechRecognizer;
import com.iflytek.cloud.SpeechUtility;
import com.iflytek.cloud.ui.RecognizerDialog;
import com.iflytek.cloud.ui.RecognizerDialogListener;

import org.json.JSONArray;
import org.json.JSONObject;
import org.json.JSONTokener;

public class MainActivity extends AppCompatActivity {

   private SpeechRecognizer mIat;
   private RecognizerDialog mIatDialog;
   private RecognizerDialogListener mRListener;

   private Button button;
   private TextView tv;
   private String result;

   @Override
   protected void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_main);

       SpeechUtility.createUtility(MainActivity.this, SpeechConstant.APPID +"=59f05070");  //=后面这里要替换成自己申请的 AppID

       mRListener = new RecognizerDialogListener() {
           @Override
           public void onResult(RecognizerResult results, boolean isLast) {
               String text = parseIatResult(results.getResultString());
               result += text;
               tv.setText(result);
               if (isLast) {

                   result = "";
              }
          }

           @Override
           public void onError(SpeechError speechError) {

          }
      };
       mIatDialog = new RecognizerDialog(MainActivity.this, null);
       mIatDialog.setListener(mRListener);

       button = (Button)findViewById(R.id.button);
       tv = (TextView)findViewById(R.id.textView);

       button.setOnClickListener(new View.OnClickListener() {
           @Override
           public void onClick(View view) {
               setIatParam("zph");
               mIatDialog.show();
          }
      });
  }

   private void setIatParam(String filename) {
       // 清空参数
       mIatDialog.setParameter(SpeechConstant.PARAMS, null);

       // 设置听写引擎
       mIatDialog.setParameter(SpeechConstant.ENGINE_TYPE, SpeechConstant.TYPE_CLOUD);

       // 设置返回结果格式
       mIatDialog.setParameter(SpeechConstant.RESULT_TYPE, "json");

       // 设置语言
       mIatDialog.setParameter(SpeechConstant.LANGUAGE, "zh_cn");
       // 设置语言区域
       mIatDialog.setParameter(SpeechConstant.ACCENT, "mandarin");

       // 设置语音前端点:静音超时时间,即用户多长时间不说话则当做超时处理
       mIatDialog.setParameter(SpeechConstant.VAD_BOS, "4000");

       // 设置语音后端点:后端点静音检测时间,即用户停止说话多长时间内即认为不再输入, 自动停止录音
       mIatDialog.setParameter(SpeechConstant.VAD_EOS, "2000");

       // 设置标点符号,设置为"0"返回结果无标点,设置为"1"返回结果有标点
       mIatDialog.setParameter(SpeechConstant.ASR_PTT, "1");

       // 设置音频保存路径,保存音频格式支持pcm、wav,设置路径为sd卡请注意WRITE_EXTERNAL_STORAGE权限
       // 注:AUDIO_FORMAT参数语记需要更新版本才能生效
       mIatDialog.setParameter(SpeechConstant.AUDIO_FORMAT,"wav");
       mIatDialog.setParameter(SpeechConstant.ASR_AUDIO_PATH, Environment.getExternalStorageDirectory() + "/MyApplication/" + filename + ".wav");
  }

   public static String parseIatResult(String json) {
       StringBuffer ret = new StringBuffer();
       try {
           JSONTokener tokener = new JSONTokener(json);
           JSONObject joResult = new JSONObject(tokener);

           JSONArray words = joResult.getJSONArray("ws");
           for (int i = 0; i < words.length(); i++) {
               // 转写结果词,默认使用第一个结果
               JSONArray items = words.getJSONObject(i).getJSONArray("cw");
               JSONObject obj = items.getJSONObject(0);
               ret.append(obj.getString("w"));
          }
      } catch (Exception e) {
           e.printStackTrace();
      }
       return ret.toString();
  }
}

Github项目地址: https://github.com/danieltangdx/xunFeiVoiceDeom

email:danieltangdx@outlook.com

讯飞SDK的使用的更多相关文章

  1. Android Studio快速集成讯飞SDK实现文字朗读功能

    今天,我们来学习一下怎么在Android Studio快速集成讯飞SDK实现文字朗读功能,先看一下效果图: 第一步 :了解TTS语音服务 TTS的全称为Text To Speech,即“从文本到语音” ...

  2. 使用讯飞SDK,实现文字在线合成语音

    private SpeechSynthesizer mTts; private int isSpeaking = 0; mTts= SpeechSynthesizer.createSynthesize ...

  3. 安卓使用讯飞sdk报错

    java.lang.NullPointerException: Attempt to invoke virtual method 'boolean com.iflytek.cloud.SpeechSy ...

  4. iphone之使用讯飞语音sdk实现语音识别功能

    1.首先下载讯飞sdk及文档:http://open.voicecloud.cn/ 2.学习里面的demo简单实现了一个小的语音识别功能 先做一个简单demo,看看识别效果.注:语音识别必须联网. 所 ...

  5. 使用javacv录像,同时进行讯飞声纹认证

    由于最近的demo中需要在活体检测的同时进行音视频录制 ,  尝试使用MediaRecord和camera来录制视频 , 然而Camera.onPreviewFrame 不能与 MediaRecord ...

  6. UNITY 接讯飞语音过程总结

    11:13 2017/3/141,安装问题:JDK与ECLIPSE位数一定要对应,32位对64位会出现 java was returned ....code 13的弹框错误.版本号可以不一致.2,EC ...

  7. 关于讯飞 使用android SDK出现21001错误码的分析

    21001,没有安装语音组件1.有没有使用SpeechUtility.createUtility()设置appid2.有没有将libmsc.so放到工程中,jar包有Msc.jar.Sunflower ...

  8. 讯飞语音SDK Android平台使用

    1. 支持功能介绍: 2. Android API主要业务接口和流程介绍 -------------------------------------------------------- 工程代码: ...

  9. 【Unity】讯飞语音识别SDK

    1.进入讯飞官网,注册帐号,进入控制台,创建新应用UnityXunfeiDemo,平台选Android.在当前应用这点下载SDK,添加AI能力(添加新服务),选择语音听写,即可下载安卓SDK(下称讯飞 ...

随机推荐

  1. [原]零基础学习SDL开发之在Android使用SDL2.0加载字体

    在上一篇文章我们知道了如何在android使用SDL2.0来渲染显示一张png图,而且在上上一篇我们知道如何使用sdl来渲染输出bmp图,那么sdl是否可以渲染输出自己喜爱的字体库的字体呢?答案是当然 ...

  2. 使用log4j让日志写入数据库

    之前做的一个项目有这么个要求,在日志管理系统里,需要将某些日志信息存储到数据库里,供用户.管理员查看分析.因此我就花了点时间搞了一下这一功能,各位请看. 摘要:我们知道log4j能提供强大的可配置的记 ...

  3. Vue-Quill-Editor插件插入图片的改进

    最近在做一个Vue-Clie小项目,使用到了Vue-Quill-Editor这个基于Vue的富文本编辑器插件.这个插件跟Vue契合良好,使用起来比其他的诸如百度UEditor要方便很多,但是存在一个小 ...

  4. bzoj2331 [SCOI2011]地板

    Description lxhgww的小名叫“小L”,这是因为他总是很喜欢L型的东西.小L家的客厅是一个的矩形,现在他想用L型的地板来铺满整个客厅,客厅里有些位置有柱子,不能铺地板.现在小L想知道,用 ...

  5. (fields.E130) DecimalFields must define a 'decimal_places' attribute.

    DecimalField类型:固定精度的十进制数,一般用来存金额相关的数据.额外的参数包括DecimalField.max_digits(整个数字的长度)和DecimalField.decimal_p ...

  6. Spring(十七)之表单处理

    表单处理在实际开发中,非常常见,比如登录.注册或者新增.修改等等. 希望本示例对于初学者有一定的提升和帮助 该表单实例,主要说明MVC,相当于前台表单提交,提交相当于一个Http请求,这个请求通过Co ...

  7. Redis口令设置

    ./redis-cli -h 192.168.128.131 -p 6379 #指定IP和端口启动对应的Redis服务 config set requirepass yourPassword #设置令 ...

  8. 【转】理解Callable 和 Spring DeferredResult

    http://www.cnblogs.com/aheizi/p/5659030.html 1-介绍 Servlet 3中的异步支持为在另一个线程中处理HTTP请求提供了可能性.当有一个长时间运行的任务 ...

  9. Spring时间(Date)类型转换+自定义

    第一种方法:利用内置的 CustomDateEditor(不推荐,每个类都需要定义) 首先,在我们的 Controller 的 InitBinder 里面,注册 CustomEditor //首先初始 ...

  10. 使用Scanner将InputStream类型转换成String

    我们在测试项目中经常会遇到这样的情形: 1. 从文件或网络得到一个InputStream,需要转换成String赋值到别的变量做为另一个方法的入参. 2. 从文件或网络得到一个InputStream后 ...