android上使用蓝牙设备进行语音输入
主要实现步骤如下:
1.确保已经和蓝牙耳机配对连接上。
2.开启蓝牙信道
AudioManager mAudioManager = (AudioManager)getSystemService(Context.AUDIO_SERVICE);
mAudioManager.setBluetoothScoOn(true);
mAudioManager.startBluetoothSco();
3.开启语音识别
4.退出时关闭蓝牙信道
mAudioManager.setBluetoothScoOn(false);
mAudioManager.stopBluetoothSco();
5.额外需要添加的权限:
<uses-permission android:name="android.permission.BROADCAST_STICKY" /> 注:部分手机如无此权限会报错
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />
以上方法到android5.0以上可能无用
提供另外一种方法
- package com.example.dkdh.testrecord;
- import android.app.Activity;
- import android.bluetooth.BluetoothAdapter;
- import android.bluetooth.BluetoothDevice;
- import android.bluetooth.BluetoothHeadset;
- import android.bluetooth.BluetoothProfile;
- import android.content.BroadcastReceiver;
- import android.content.Context;
- import android.content.Intent;
- import android.content.IntentFilter;
- import android.media.AudioManager;
- import android.media.MediaPlayer;
- import android.media.MediaRecorder;
- import android.os.Bundle;
- import android.os.Environment;
- import android.util.Log;
- import android.view.View;
- import android.widget.Button;
- import android.widget.TextView;
- import android.widget.Toast;
- import com.example.dkdh.testrecord.util.FucUtil;
- import com.example.dkdh.testrecord.util.JsonParser;
- import com.iflytek.cloud.InitListener;
- import com.iflytek.cloud.RecognizerListener;
- 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.ErrorCode;
- import java.io.IOException;
- import java.util.HashMap;
- import java.util.LinkedHashMap;
- import java.util.List;
- public class MainActivity extends Activity implements View.OnClickListener{
- private final String TAG = MainActivity.class.getSimpleName();
- private final String XF_APP_ID = "xxxxxx";
- BluetoothHeadset headset;
- private Button start,stop;
- private TextView result;
- private AudioManager mAudioManager = null;
- private BluetoothHeadset mBluetoothHeadset;
- private BluetoothAdapter mBluetoothAdapter;
- private BluetoothDevice mBluetoothDevice;
- private SpeechRecognizer mIat;
- // // 语音听写UI
- // private RecognizerDialog mIatDialog;
- // // 用HashMap存储听写结果
- private HashMap<String, String> mIatResults = new LinkedHashMap<String, String>();
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- SpeechUtility.createUtility(this, SpeechConstant.APPID + "=" + XF_APP_ID);
- result = (TextView)findViewById(R.id.result);
- start = (Button)findViewById(R.id.startRec);
- stop = (Button)findViewById(R.id.stopRec);
- start.setOnClickListener(this);
- stop.setOnClickListener(this);
- mAudioManager = (AudioManager)getSystemService(Context.AUDIO_SERVICE);
- mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
- mBluetoothAdapter.getProfileProxy(this, mProfileListener, BluetoothProfile.HEADSET);
- // // 初始化识别无UI识别对象
- // // 使用SpeechRecognizer对象,可根据回调消息自定义界面;第二个参数:本地识别时传mInitListener
- mIat = SpeechRecognizer.createRecognizer(this, mInitListener);
- }
- private BluetoothProfile.ServiceListener mProfileListener = new BluetoothProfile.ServiceListener(){
- @Override
- public void onServiceConnected(int profile, BluetoothProfile proxy) {
- if (profile == BluetoothProfile.HEADSET){
- mBluetoothHeadset = (BluetoothHeadset) proxy;
- List<BluetoothDevice> devices = mBluetoothHeadset.getConnectedDevices();
- if (devices.size()>0){
- mBluetoothDevice = devices.get(0);
- int state = mBluetoothHeadset.getConnectionState(mBluetoothDevice);
- Log.e("==============","headset state:"+state);
- if (state==BluetoothHeadset.STATE_CONNECTED){
- Log.e("=================","bluetooth headset connected");
- }
- }
- }
- }
- @Override
- public void onServiceDisconnected(int profile) {
- if (profile == BluetoothProfile.HEADSET){
- mBluetoothHeadset = null;
- }
- }
- };
- @Override
- public void onClick(View v) {
- switch (v.getId()){
- case R.id.startRec:
- startRecordWav();
- break;
- case R.id.stopRec:
- stopRecordWav();
- break;
- default:
- break;
- }
- }
- /**
- * 识别实时语音
- */
- private void recog(){
- mIatResults.clear();
- // 设置参数
- setParam();
- int ret = 0;
- // 不显示听写对话框
- ret = mIat.startListening(mRecognizerListener);
- if (ret != ErrorCode.SUCCESS) {
- showTip("听写失败,错误码:" + ret);
- } else {
- // showTip("");
- }
- }
- /**
- * 初始化监听器。
- */
- private InitListener mInitListener = new InitListener() {
- @Override
- public void onInit(int code) {
- Log.i(TAG, "SpeechRecognizer init() code = " + code);
- if (code != ErrorCode.SUCCESS) {
- showTip("初始化失败,错误码:" + code);
- }
- }
- };
- /**
- * 听写监听器。
- */
- private RecognizerListener mRecognizerListener = new RecognizerListener() {
- @Override
- public void onBeginOfSpeech() {
- // 此回调表示:sdk内部录音机已经准备好了,用户可以开始语音输入
- showTip("开始说话");
- }
- @Override
- public void onError(SpeechError error) {
- // Tips:
- // 错误码:10118(您没有说话),可能是录音机权限被禁,需要提示用户打开应用的录音权限。
- showTip(error.getPlainDescription(true));
- // showTip("错误码:10118(您没有说话),可能是录音机权限被禁,请打开应用的录音权限。");
- }
- @Override
- public void onEndOfSpeech() {
- // 此回调表示:检测到了语音的尾端点,已经进入识别过程,不再接受语音输入
- showTip("结束说话");
- }
- @Override
- public void onResult(RecognizerResult results, boolean isLast) {
- String text = JsonParser.parseIatResult(results.getResultString());
- Log.i(TAG, text);
- showTip(text);
- if(isLast) {
- //TODO 最后的结果
- result.append(text);
- }
- }
- @Override
- public void onVolumeChanged(int volume, byte[] data) {
- // showTip("当前正在说话,音量大小:" + volume);
- Log.i(TAG,"当前正在说话,音量大小:" + volume);
- Log.i(TAG, "返回音频数据:" + data.length);
- }
- @Override
- public void onEvent(int eventType, int arg1, int arg2, Bundle obj) {
- // 以下代码用于获取与云端的会话id,当业务出错时将会话id提供给技术支持人员,可用于查询会话日志,定位出错原因
- // 若使用本地能力,会话id为null
- // if (SpeechEvent.EVENT_SESSION_ID == eventType) {
- // String sid = obj.getString(SpeechEvent.KEY_EVENT_SESSION_ID);
- // Log.d(TAG, "session id =" + sid);
- // }
- }
- };
- /**
- * Toast显示提示
- */
- private void showTip(String str){
- Toast.makeText(this,str,Toast.LENGTH_SHORT).show();
- }
- /**
- * 参数设置
- * @return
- */
- public void setParam(){
- // 清空参数
- mIat.setParameter(SpeechConstant.PARAMS, null);
- // 设置引擎
- mIat.setParameter(SpeechConstant.ENGINE_TYPE, SpeechConstant.TYPE_CLOUD);
- // 设置返回结果格式
- mIat.setParameter(SpeechConstant.RESULT_TYPE, "json");
- // 设置语言
- mIat.setParameter(SpeechConstant.LANGUAGE, "zh_cn");
- // 设置语言区域
- mIat.setParameter(SpeechConstant.ACCENT,"mandarin");
- // 设置语音前端点:静音超时时间,即用户多长时间不说话则当做超时处理
- mIat.setParameter(SpeechConstant.VAD_BOS, "8000");
- // 设置语音后端点:后端点静音检测时间,即用户停止说话多长时间内即认为不再输入, 自动停止录音
- mIat.setParameter(SpeechConstant.VAD_EOS, "1000");
- // 设置标点符号,设置为"0"返回结果无标点,设置为"1"返回结果有标点
- mIat.setParameter(SpeechConstant.ASR_PTT, "0");
- // 设置音频保存路径,保存音频格式支持pcm、wav,设置路径为sd卡请注意WRITE_EXTERNAL_STORAGE权限
- // 注:AUDIO_FORMAT参数语记需要更新版本才能生效
- mIat.setParameter(SpeechConstant.AUDIO_FORMAT,"wav");
- mIat.setParameter(SpeechConstant.ASR_AUDIO_PATH, Environment.getExternalStorageDirectory()+"/msc/iat.wav");
- // 设置录音时长,单位ms
- mIat.setParameter(SpeechConstant.KEY_SPEECH_TIMEOUT,"60000");
- //设置音频源,MIC、VOICE_RECOGNITION、VOICE_COMMUNICATION可用,但与不同android系统有关
- mIat.setParameter(SpeechConstant.AUDIO_SOURCE, MediaRecorder.AudioSource.VOICE_COMMUNICATION+"");
- }
- /**
- * 停止录音
- */
- private void stopRecordWav(){
- Log.e(TAG, "停止录音");
- mBluetoothHeadset.stopVoiceRecognition(mBluetoothDevice);
- }
- /**
- * 录音,自主控制录音格式、速率等
- */
- private void startRecordWav(final int source){
- if (!mAudioManager.isBluetoothScoAvailableOffCall()) {
- Log.d(TAG, "系统不支持蓝牙录音");
- return;
- }
- if (mBluetoothHeadset == null){
- Toast.makeText(this, "蓝牙耳机对象null",Toast.LENGTH_SHORT).show();
- return;
- }
- if (mBluetoothDevice!=null){
- mBluetoothHeadset.startVoiceRecognition(mBluetoothDevice);
- }
- IntentFilter audioStateFilter = new IntentFilter();
- audioStateFilter.addAction(BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED);
- audioStateFilter.addAction(BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED);
- registerReceiver(new BroadcastReceiver() {
- @Override
- public void onReceive(Context context, Intent intent) {
- String action = intent.getAction();
- if (action.equals(BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED)){
- int state = intent.getIntExtra(BluetoothProfile.EXTRA_STATE,-1);
- if (state==BluetoothHeadset.STATE_AUDIO_CONNECTED){
- Log.e("==============","开始蓝牙语音识别");
- recog();
- unregisterReceiver(this); // 别遗漏
- }
- }
- }
- },audioStateFilter);
- }
- @Override
- protected void onResume(){
- super.onResume();
- }
- }
转载于:https://www.cnblogs.com/dongweiq/p/8288336.html
android上使用蓝牙设备进行语音输入相关推荐
- android手机设置中的语音输入与输出
android手机设置中的语音输入与输出: 设置--语音输入与输出--文字转语音设置--安装语音数据库 在电子市场下载安装,安装成功后,将键盘改为Android键盘,就会看到一个麦克风一样的图标,点击 ...
- android 输入法,里面还集成语音输入
<?xml version="1.0" encoding="utf-8"?> <com.example.android.softkeyboar ...
- android录音波浪动画_Android语音输入的波浪效果 – WaveView
效果展示 源码地址 实现思路 绘制正弦波形 水平移动波形,即可得到破浪效果. 正弦波形的绘制 private void createShader() { ... Bitmap bitmap = Bit ...
- html5 语音输入小话筒,HTML5语音输入方法
谷歌的网站是时逛时新啊,今天在他们首页发现了HTML5的新玩法--语音搜索.可惜的是只有webkit核心的浏览器才能使用.用法很简单 只需要在input添加属性 x-webkit-speech 即可, ...
- 手机百度输入法环境:android 1.6,百度手机输入法Android 5.1版—新增粤语语音输入...
近日,百度手机输入法发布了全新安卓5.1版,主打离线语音.粤语语音.智能英文.英文单词手写.全新皮肤等功能,获得了不少用户的好评.据了解,百度手机输入法安卓5.1版在原5.0版简约风格的基础上更加专注 ...
- Android 蓝牙耳机 语音输入与播放
Android 蓝牙耳机 语音输入与播放 原以为手机连上蓝牙耳机就能录入语音信号,too young to simple. 经过一番搜寻与折腾,找到两种方式: AudioManager.startBl ...
- 教你用 Android 做二次开发,识别率达到科大讯飞语音输入水平 | 原力计划
作者 | Pek_KuaiJia 责编 | 夕颜 头图 | CSDN 下载自视觉中国 出品 | CSDN(ID:CSDNnews) 随着目前用户需求的精细化和智能化,很多时候我们需要在App内集成语音 ...
- android语音输入文字,盘点好用的语音输入APP,懒得打字的时候就说话吧!
原标题:盘点好用的语音输入APP,懒得打字的时候就说话吧! 本文为「智活范」原创作品,欢迎关注我们! 上次推完好用的录音APP后,立刻就有萌友来问了,能不能直接录音转文字呢,这样说话就能生成文字,多省 ...
- Android 语音输入API使用
Android 语音输入API使用 转载http://www.javaarch.net/jiagoushi/782.htm Android已经支持语音输入的API了,不过不知道中文输入识别效果怎么样. ...
最新文章
- Error in sort.int(x, na.last = na.last, decreasing = decreasing, ...) : ‘x‘ must be atomic
- 研究揭示大脑在工作记忆中存储信息的神经机制
- CF359D:Pair of Numbers(数论)
- java线程callback,Java线程之异步回调(Callback)
- python常用内置模块-Python之OS模块常用内置方法汇总
- python之socket
- 题目1179:阶乘-------------阶乘不用long long int 就不能AC
- ligerUI的列头合并代码片段
- openshift_云上的播放框架变得简单:Openshift模块
- 198. house robber 题解
- python 操作excel神器_【转】多图+代码 | 详解Python操作Excel神器openpyxl的各种操作!...
- 每天一个linux命令(30):cal 命令
- ubuntu16.04 运行dso问题梳理
- SQL创建数据库– PostgreSQL,MySQL,SQL Server
- 缓存与缓冲的区别 Difference Between Cache and Buffer
- 软件系统测试流程规范
- 美通企业日报 | 乐高在中国大陆已开设100家门店;香港首批虚拟银行的成立如箭在弦...
- 疫情下技术人的宅家指南
- html tooltips效果,html5tooltips.js – 一款轻量级的3D工具提示插件
- topic是短语还是句子_新仁爱版英语unit4 topic1短语和句子
热门文章
- 360脱壳-native函数还原笔记-2017-06-25
- Android6.0源码分析—— Zygote进程分析(补充)
- Android framwork service添加(manager 远程调service,service jni调native code)
- Android深入透析之常用设计模式经验谈
- html.编辑数据回显,从HTML表格编辑/更新MySQL数据库值
- 全国计算机二级qq闪退,电脑上QQ闪退怎么回事?各个系统版本电脑QQ闪退现象的解决方法介绍...
- 大数据可视化html模板开源_让数据栩栩如生,蚂蚁金服新一代开源数据可视化解决方案——AntV...
- 关于 ST mems传感器开发的一些技术分享
- webstorm怎么跑项目_怎么跑Mint-UI的实例,你知道吗?
- 2020-07-08 CVPR2020 表示学习论文讨论(3) 笔记