Android使用AudioRecord判断是否有音频输入
Android党都应该玩过一个叫吹裙子的游戏,这个游戏就是原理就是通过监听用户吹出的气的力度来决定如何把MM的裙子弄飞起来的,所以关键在于如何判断用户吹气的力度问题。现在公司刚好有这个需求要评估就是需要一直监听用户的语音输入,当在两秒内没有语音输入时候就暂停某项事情,有语音输入的时候就要继续做某件事。其实这两件事情的本质原理是一样的,就是通过这AudioRecord来处理用户输入的原始音频数据,从而计算出当前用户输入的音量大小来判断用户是否有语音输入。下面贴上一段代码用户可以自行研究。
代码如下 | 复制代码 |
publicstaticintcalculateVolume(byte[] var0,intvar1) { int[] var3 =null; intvar4 = var0.length; intvar2; if(var1 ==8) { var3 =newint[var4]; for(var2 =0; var2 < var4; ++var2) { var3[var2] = var0[var2]; } }elseif(var1 ==16) { var3 =newint[var4 /2]; for(var2 =0; var2 < var4 /2; ++var2) { bytevar5 = var0[var2 *2]; bytevar6 = var0[var2 *2+1]; intvar13; if(var5<0) { var13 = var5 +256; }else{ var13 = var5; } shortvar7 = (short)(var13 +0); if(var6<0) { var13 = var6 +256; }else{ var13 = var6; } var3[var2] = (short)(var7 + (var13 <<8)); } } int[] var8 = var3; if(var3 !=null&& var3.length !=0) { floatvar10 =0.0F; for(intvar11 =0; var11 < var8.length; ++var11) { var10 += (float)(var8[var11] * var8[var11]); } var10 /= (float)var8.length; floatvar12 =0.0F; for(var4 =0; var4 < var8.length; ++var4) { var12 += (float)var8[var4]; } var12 /= (float)var8.length; var4 = (int)(Math.pow(2.0D, (double)(var1 -1)) -1.0D); doublevar14 = Math.sqrt((double)(var10 - var12 * var12)); intvar9; if((var9 = (int)(10.0D * Math.log10(var14 *10.0D * Math.sqrt(2.0D) / (double)var4 +1.0D)))<0) { var9 =0; } if(var9 >10) { var9 =10; } returnvar9; }else{ return0; } } |
上述方法的形参中的一个字节数组便是当前捕捉到用户输入的原始音频数据,第二个形参则是音频数据格式位PCM,16位每个样本,比如如果音频数据格式设为AudioFormat.ENCODING_PCM_16BIT,则参数传16。最后实现的效果就是如下:
01-12 18:54:17.222 E/recordtest(24249): no voice input
…………….
01-12 18:54:19.027 E/recordtest(24249): no voice input
01-12 18:54:19.067 E/recordtest(24249): no voice input
01-12 18:54:19.105 E/recordtest(24249): no voice input
01-12 18:54:19.145 E/recordtest(24249): no voice input
01-12 18:54:19.188 E/recordtest(24249): no voice input
01-12 18:54:19.227 E/recordtest(24249): no voice input
01-12 18:54:19.228 E/recordtest(24249): no voice input in 2 sec
01-12 18:54:19.269 E/recordtest(24249): no voice input
01-12 18:54:19.306 E/recordtest(24249): no voice input
01-12 18:54:19.345 E/recordtest(24249): no voice input
01-12 18:54:19.387 E/recordtest(24249): no voice input
01-12 18:54:19.427 E/recordtest(24249): no voice input
01-12 18:54:19.466 E/recordtest(24249): has voice input
01-12 18:54:19.505 E/recordtest(24249): has voice input
01-12 18:54:19.546 E/recordtest(24249): has voice input
01-12 18:54:19.586 E/recordtest(24249): has voice input
01-12 18:54:19.646 E/recordtest(24249): has voice input
01-12 18:54:19.686 E/recordtest(24249): has voice input
01-12 18:54:19.726 E/recordtest(24249): has voice input
01-12 18:54:19.766 E/recordtest(24249): has voice input
代码如下 | 复制代码 |
//activity的xml <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="baway.com.cj.myrecyclerview2.MainActivity"> <android.support.v4.widget.SwipeRefreshLayout android:id="@+id/swipeRefreshLayout" android:layout_width="match_parent" android:layout_above="@+id/progress" android:layout_height="match_parent"> <android.support.v7.widget.RecyclerView android:id="@+id/recyclerView" android:layout_width="match_parent" android:layout_height="match_parent">
<ProgressBar android:id="@+id/progress" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:visibility="gone" android:layout_alignParentBottom="true"/> item1的xml和item2的xml就一个背景颜色不同 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#f00000" tools:context="baway.com.cj.myrecyclerview2.MainActivity"> <TextView android:id="@+id/name" android:layout_width="match_parent" android:layout_height="50dp" android:textSize="20sp" android:gravity="center" android:textColor="#00ff00"/> item2 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#f000ff" tools:context="baway.com.cj.myrecyclerview2.MainActivity"> <TextView android:id="@+id/name" android:layout_width="match_parent" android:layout_height="50dp" android:textSize="20sp" android:gravity="center" android:textColor="#00ff00"/> //activity,有上啦下拉,添加分割线, packagebaway.com.cj.myrecyclerview2; importandroid.graphics.Color; importandroid.os.Handler; importandroid.support.v4.widget.SwipeRefreshLayout; importandroid.support.v7.app.AppCompatActivity; importandroid.os.Bundle; importandroid.support.v7.widget.DefaultItemAnimator; importandroid.support.v7.widget.LinearLayoutManager; importandroid.support.v7.widget.RecyclerView; importandroid.util.Log; importandroid.view.View; importandroid.widget.ProgressBar; importandroid.widget.Toast; importjava.util.ArrayList; publicclassMainActivityextendsAppCompatActivity { privateRecyclerView recyclerView; privateMainAdapter mainAdapter; privateSwipeRefreshLayout swipeRefreshLayout; privateLinearLayoutManager linearLayoutManager; privateProgressBar progress; @Override protectedvoidonCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initView(); initData(); } privatevoidinitView() { recyclerView = (RecyclerView)findViewById(R.id.recyclerView); swipeRefreshLayout = (SwipeRefreshLayout) findViewById(R.id.swipeRefreshLayout); progress = (ProgressBar)findViewById(R.id.progress); linearLayoutManager =newLinearLayoutManager(this); recyclerView.setLayoutManager(linearLayoutManager); mainAdapter =newMainAdapter(this); mainAdapter.setOnClickListener(newMyItemOnClickListener() { @Override publicvoidonItemClick(View view,intposition) { Toast.makeText(MainActivity.this," "+position, Toast.LENGTH_SHORT).show(); } }); recyclerView.setAdapter(mainAdapter); recyclerView.addItemDecoration(newMyItemDecoration()); recyclerView.setItemAnimator(newDefaultItemAnimator()); swipeRefreshLayout.setColorSchemeColors(Color.RED,Color.GREEN,Color.BLUE); swipeRefreshLayout.setProgressViewOffset(true,0,200); swipeRefreshLayout.setOnRefreshListener(newSwipeRefreshLayout.OnRefreshListener() { @Override publicvoidonRefresh() { newHandler().postDelayed(newRunnable() { @Override publicvoidrun() { swipeRefreshLayout.setRefreshing(false); } },3000); } }); recyclerView.addOnScrollListener(newRecyclerView.OnScrollListener() { @Override publicvoidonScrolled(RecyclerView recyclerView,intdx,intdy) { super.onScrolled(recyclerView, dx, dy); intlastVisibleItemPosition = linearLayoutManager.findLastVisibleItemPosition(); intitemCount = linearLayoutManager.getItemCount(); if(lastVisibleItemPosition +1>= itemCount && dy >0) { progress.setVisibility(View.VISIBLE); newHandler().postDelayed(newRunnable() { @Override publicvoidrun() { progress.setVisibility(View.GONE); } },3000); } } @Override publicvoidonScrollStateChanged(RecyclerView recyclerView,intnewState) { super.onScrollStateChanged(recyclerView, newState); } }); } privatevoidinitData() { ArrayList for(inti =0; i<50; i++) { strings.add("name "+i); } mainAdapter.setData(strings); mainAdapter.notifyDataSetChanged(); } } //adpter有多条目展示 packagebaway.com.cj.myrecyclerview2; importandroid.content.Context; importandroid.support.v7.widget.RecyclerView; importandroid.view.LayoutInflater; importandroid.view.View; importandroid.view.ViewGroup; importandroid.widget.TextView; importjava.util.ArrayList; /** * Created by cj on 2017/1/12. */ publicclassMainAdapterextendsRecyclerView.Adapter { privateContext context; privateArrayList privateMyItemOnClickListener myItemOnClickListener; privatefinalintTYPE_0 =0; privatefinalintTYPE_1 =1; publicMainAdapter(Context context) { this.context = context; } @Override publicintgetItemViewType(intposition) { if(position %3==0) { returnTYPE_0; }else{ returnTYPE_1; } } publicvoidsetOnClickListener(MyItemOnClickListener onClickListener) { this.myItemOnClickListener = onClickListener; } @Override publicRecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent,intviewType) { if(viewType == TYPE_0) { View inflate = LayoutInflater.from(context).inflate(R.layout.item_main,null); returnnewMyViewHolder(inflate); }else{ View inflate = LayoutInflater.from(context).inflate(R.layout.item_main2,null); returnnewMyViewHolder2(inflate); } } @Override publicvoidonBindViewHolder(RecyclerView.ViewHolder holder,finalintposition) { if(getItemViewType(position) == TYPE_0) { MyViewHolder myViewHolder = (MyViewHolder)holder; myViewHolder.name.setText(datas.get(position)); myViewHolder.name.setOnClickListener(newView.OnClickListener() { @Override publicvoidonClick(View v) { myItemOnClickListener.onItemClick(v,position); } }); }else{ MyViewHolder2 myViewHolder = (MyViewHolder2)holder; myViewHolder.name.setText(datas.get(position)); myViewHolder.name.setOnClickListener(newView.OnClickListener() { @Override publicvoidonClick(View v) { myItemOnClickListener.onItemClick(v,position); } }); } } @Override publicintgetItemCount() { returndatas.size(); } publicvoidsetData(ArrayList if(data ==null) { return; } this.datas.addAll(data); } publicclassMyViewHolderextendsRecyclerView.ViewHolder{ publicTextView name; publicMyViewHolder(View itemView) { super(itemView); name = (TextView) itemView.findViewById(R.id.name); } } publicclassMyViewHolder2extendsRecyclerView.ViewHolder{ publicTextView name; publicMyViewHolder2(View itemView) { super(itemView); name = (TextView) itemView.findViewById(R.id.name); } } } //自定义分割线继承系统的 RecyclerView.ItemDecoration packagebaway.com.cj.myrecyclerview2; importandroid.graphics.Canvas; importandroid.graphics.Color; importandroid.graphics.Paint; importandroid.graphics.Rect; importandroid.support.v7.widget.RecyclerView; importandroid.util.Log; importandroid.view.View; /** * Created by cj on 2017/1/12. */ publicclassMyItemDecorationextendsRecyclerView.ItemDecoration { privatePaint paint; publicMyItemDecoration() { paint =newPaint(); paint.setColor(Color.BLUE); } @Override publicvoidonDraw(Canvas c, RecyclerView parent, RecyclerView.State state) { super.onDraw(c, parent, state); //c.drawCircle(300,300,300,paint); intchildCount = parent.getChildCount(); for(inti =0; i < childCount; i++) { View childAt = parent.getChildAt(i); c.drawRect(childAt.getLeft(),childAt.getBottom(),childAt.getWidth()+childAt.getLeft(),childAt.getBottom()+5,paint); } } @Override publicvoidonDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) { super.onDrawOver(c, parent, state); // int childCount = parent.getChildCount(); // for (int i = 0; i < childCount; i++) { // View childAt = parent.getChildAt(i); // c.drawRect(childAt.getLeft(),childAt.getTop(),childAt.getWidth()+childAt.getLeft(),childAt.getBottom()+5,paint); // } } @Override publicvoidgetItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { super.getItemOffsets(outRect, view, parent, state); outRect.set(20,0,20,5); } } |
相关文章
- 有时为了网站安全和版权问题,会对自己写的php源码进行加密,在php加密技术上最常用的是zend公司的zend guard 加密软件,现在我们来图文讲解一下。 下面就简单说说如何...2016-11-25
- 本篇文章主要分享了通过window.navigator来判断浏览器及其版本信息的实例代码。具有一定的参考价值,下面跟着小编一起来看下吧...2017-01-23
- 下面我们来看一篇关于Android子控件超出父控件的范围显示出来方法,希望这篇文章能够帮助到各位朋友,有碰到此问题的朋友可以进来看看哦。 <RelativeLayout xmlns:an...2016-10-02
- ps软件是现在很多人都会使用到的,HSL面板在ps软件中又有着非常独特的作用。这次文章就给大家介绍下ps怎么使用HSL面板,还不知道使用方法的下面一起来看看。  ...2017-07-06
- 许多的朋友对于Plesk控制面板应用不是非常的了解特别是英文版的Plesk控制面板,在这里小编整理了一些关于Plesk控制面板常用的使用方案整理,具体如下。 本文基于Linu...2016-10-10
使用insertAfter()方法在现有元素后添加一个新元素
复制代码 代码如下: //在现有元素后添加一个新元素 function insertAfter(newElement, targetElement){ var parent = targetElement.parentNode; if (parent.lastChild == targetElement){ parent.appendChild(newEl...2014-05-31Android开发中findViewById()函数用法与简化
findViewById方法在android开发中是获取页面控件的值了,有没有发现我们一个页面控件多了会反复研究写findViewById呢,下面我们一起来看它的简化方法。 Android中Fin...2016-09-20- 使用 conditional comment 来判断 IE 的版本。嗯,是早早有人提出,但没有认真看代码。昨天刚好在看 CSS3 PIE 的时候看到,觉得是不是不靠谱。今天看到 Paul Irish 也提起,那么,推荐一下吧。这是作者博客上写的:复制代码 代码...2014-05-31
- 如果我们的项目需要做来电及短信的功能,那么我们就得在Android模拟器开发这些功能,本来就来告诉我们如何在Android模拟器上模拟来电及来短信的功能。 在Android模拟...2016-09-20
使用percona-toolkit操作MySQL的实用命令小结
1.pt-archiver 功能介绍: 将mysql数据库中表的记录归档到另外一个表或者文件 用法介绍: pt-archiver [OPTION...] --source DSN --where WHERE 这个工具只是归档旧的数据,不会对线上数据的OLTP查询造成太大影响,你可以将...2015-11-24- 大概有如下步骤 新建项目Bejs 新建文件package.json 新建文件Gruntfile.js 命令行执行grunt任务 一、新建项目Bejs源码放在src下,该目录有两个js文件,selector.js和ajax.js。编译后代码放在dest,这个grunt会...2014-06-07
如何使用php脚本给html中引用的js和css路径打上版本号
在搜索引擎中搜索关键字.htaccess 缓存,你可以搜索到很多关于设置网站文件缓存的教程,通过设置可以将css、js等不太经常更新的文件缓存在浏览器端,这样访客每次访问你的网站的时候,浏览器就可以从浏览器的缓存中获取css、...2015-11-24- 下面小编就为大家分享一篇c# 判断是否为空然后赋值的4种实现方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-06-25
- 夜神android模拟器如何设置代理呢?对于这个问题其实操作起来是非常的简单,下面小编来为各位详细介绍夜神android模拟器设置代理的方法,希望例子能够帮助到各位。 app...2016-09-20
jQuery 1.9使用$.support替代$.browser的使用方法
jQuery 从 1.9 版开始,移除了 $.browser 和 $.browser.version , 取而代之的是 $.support 。 在更新的 2.0 版本中,将不再支持 IE 6/7/8。 以后,如果用户需要支持 IE 6/7/8,只能使用 jQuery 1.9。 如果要全面支持 IE,并混合...2014-05-31- 为了增强android应用的用户体验,我们可以在一些Button按钮上自定义动态的设置一些样式,比如交互时改变字体、颜色、背景图等。 今天来看一个通过重写Button来动态实...2016-09-20
- 如果我们要在Android应用APP中加载html5页面,我们可以使用WebView,本文我们分享两个WebView加载html5页面实例应用。 实例一:WebView加载html5实现炫酷引导页面大多...2016-09-20
安装和使用percona-toolkit来辅助操作MySQL的基本教程
一、percona-toolkit简介 percona-toolkit是一组高级命令行工具的集合,用来执行各种通过手工执行非常复杂和麻烦的mysql和系统任务,这些任务包括: 检查master和slave数据的一致性 有效地对记录进行归档 查找重复的索...2015-11-24- C#注释的一些使用方法浅谈,需要的朋友可以参考一下...2020-06-25
- 深入理解Android中View和ViewGroup从组成架构上看,似乎ViewGroup在View之上,View需要继承ViewGroup,但实际上不是这样的。View是基类,ViewGroup是它的子类。本教程我们深...2016-09-20