【第22期】观点:IT 行业加班,到底有没有价值?

安卓手机麦克风测试应用开发

原创 2015年11月17日 16:57:38

        之前接了一个开发测试手机麦克风寿命apk的任务,主要要求是:1、可以控制手机麦克风的开启与关闭;2、麦克风接收的声音从听筒发出;3、可设置测试时间。下面我对该应用的开发做一相关讲解并附上代码。

        开发难点:1、在话筒与听筒均开启的情况下,进入话筒的声音无法直接从听筒发出,需要做相关转换输出处理;

                            2、时间任务的启动与停止;

                            3、时间设置与数据类型的转换。

        解决方法:1、引入音频缓冲区作为音频接收与播放的中专(Buffer),以音频声音作为对象由缓冲区接收存储(音频录制),以缓冲区的对象作为播放对象由听筒播放出来;

                            2、将麦克风开启方法以线程体来定义和调用;

                            3、以EditText作为时间输入的控件来设置测试时长,并前值转换为长整型。

以下是代码:

主类:AndroidMikeTestActivity.java

package com.cfzz.amt;

import com.android.methed.Run_Methed;


import android.os.Bundle;
import android.app.Activity;
import android.view.KeyEvent;
import android.view.View;
import android.widget.Button;
import android.view.View.OnClickListener;
import android.widget.EditText;
import android.widget.TextView;
import android.view.WindowManager;
import android.os.SystemClock;
import android.widget.Chronometer;
import java.util.Timer;
import java.util.TimerTask;

public class AndroidMikeTestActivity extends Activity implements
		OnClickListener {
	Run_Methed run_Metned = new Run_Methed();
	Thread thread = null;
	private EditText timelength;
	Button playbutton, stopbutton;
	private Timer timer;
	private Chronometer time;
	private long recordingTime = 0;

	public void onRecordStart()// time start
	{
		time.setBase(SystemClock.elapsedRealtime() - recordingTime);// 跳过已经记录了的时间,起到继续计时的作用
		time.start();
	}

	public void onRecordPause()// time pause
	{
		time.stop();
		recordingTime = SystemClock.elapsedRealtime() - time.getBase();// 保存这次记录了的时间
	}

	public void onRecordStop()// time stop
	{
		recordingTime = 0;
		time.setBase(SystemClock.elapsedRealtime());
	}

	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		timelength = (EditText) this.findViewById(R.id.timelength);
		timelength.setText("0");
		time = (Chronometer) this.findViewById(R.id.chronometer);//引入计时器控件
		time.setBase(SystemClock.elapsedRealtime());//初始化时间
		// 按键监听
		playbutton = (Button) findViewById(R.id.playbutton);
		stopbutton = (Button) findViewById(R.id.stopbutton);
		playbutton.setOnClickListener(this);
		stopbutton.setOnClickListener(this);
	}

	public void onClick(View v) {
		switch (v.getId()) {
		case R.id.playbutton:
			// 按钮【开】
			long t = Long.parseLong("" + timelength.getText());//以小时为单位设置时间
			if (t != 0) {

				Thread thread = new Thread(run_Metned);//将麦克风开启方法加入线程体中
				thread.start();
				onRecordStart();
				long s = 1000 * 3600 * t ;//将小时数转化为毫秒数
				timer = new Timer(true);

				TimerTask task = new TimerTask()// 计时任务设置
				{
					public void run() {
						run_Metned.no();
						onRecordPause();
						timer.cancel();

					}
				};
				timer.schedule(task, s, 1);//s毫秒后停止
			} else if (t == 0) {
				run_Metned.no();
			} else if (timelength.getText() == null) {
				run_Metned.no();
				timelength.setText("0");
			}
			break;
		case R.id.stopbutton:
			// 按钮【关】
			run_Metned.no();
			timer.cancel();
			onRecordStop();
			onRecordPause();
			timelength.setText("0");
			break;
		}
	}

	int i = 0;

	public boolean onKeyDown(int keyCode, KeyEvent event) //双击返回键退出
        {

		if (keyCode == KeyEvent.KEYCODE_BACK
				&& event.getAction() == KeyEvent.ACTION_DOWN) {
			long t = Long.parseLong("" + timelength.getText());
			i++;
			if (i == 2 && t != 0) {
				run_Metned.no();
				timer.cancel();
				onRecordStop();
				onRecordPause();
			}
			if (i == 3) {
				finish();
			}

		}
		return true;
	}

	/*
	 * @Override protected void onDestroy() {
	 * 
	 * thread.interrupt(); run_Metned.no(); timer.cancel(); onRecordStop();
	 * super.onDestroy(); }
	 */
}
方法类:

Run_Methed.java

package com.android.methed;

import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.AudioTrack;
import android.media.MediaRecorder;

public class Run_Methed implements Runnable {
	// 是否录放的标记
	boolean isRecording = false;
	static final int frequency = 44100;
	@SuppressWarnings("deprecation")
	static final int channelConfiguration = AudioFormat.CHANNEL_CONFIGURATION_MONO;
	static final int audioEncoding = AudioFormat.ENCODING_PCM_16BIT;
	// 定义录放音缓冲区大小
	int recBufSize, playBufSize;
	// 实例化录音对象
	AudioRecord audioRecord;
	// 实例化播放对象
	AudioTrack audioTrack;

	public Run_Methed() {
		// 调用getMinBufferSize方法获得录音的最小缓冲空间
		recBufSize = AudioRecord.getMinBufferSize(frequency,
				channelConfiguration, audioEncoding);

		// 调用getMinBufferSize方法获得放音最小的缓冲区大小
		playBufSize = AudioTrack.getMinBufferSize(frequency,
				channelConfiguration, audioEncoding);
		// -----------------------------------------、
		// 调用构造函数实例化录音对象
		audioRecord = new AudioRecord(MediaRecorder.AudioSource.MIC, frequency,
				channelConfiguration, audioEncoding, recBufSize);

		// 调用构造函数实例化放音对象,以听筒模式播放
		audioTrack = new AudioTrack(AudioManager.<span style="background-color: rgb(51, 204, 255);">STREAM_VOICE_CALL</span>, frequency,
				channelConfiguration, audioEncoding, playBufSize,
				AudioTrack.MODE_STREAM);
	}

	// 线程体
	public void run() {

		try {
			byte[] buffer = new byte[recBufSize];
			audioRecord.startRecording();// 开始录制
			audioTrack.play();// 开始播放
			this.isRecording = true;
			while (isRecording) {
				// 从MIC保存数据到缓冲区
				int bufferReadResult = audioRecord.read(buffer, 0, recBufSize);

				byte[] tmpBuf = new byte[bufferReadResult];
				System.arraycopy(buffer, 0, tmpBuf, 0, bufferReadResult);
				// 写入数据就播放
				audioTrack.write(tmpBuf, 0, tmpBuf.length);
			}
			audioTrack.stop();
			audioRecord.stop();
		} catch (Throwable t) {
		}
	}

	// 停止方法
	public void no() {
		isRecording = false;
	}

}

Track_Methed,java

package com.android.methed;

import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioTrack;

public class Track_Methed implements Runnable {

	// 播放暂存
	int out_size;
	// 放音对象
	AudioTrack track = null;

	public void run() {

	}

	// 播送方法
	public void track() {
		out_size = android.media.AudioTrack.getMinBufferSize(8000,
				AudioFormat.CHANNEL_CONFIGURATION_MONO,
				AudioFormat.ENCODING_PCM_16BIT);

		track = new AudioTrack(AudioManager.STREAM_VOICE_CALL, 8000,
				AudioFormat.CHANNEL_CONFIGURATION_MONO,
				AudioFormat.ENCODING_PCM_16BIT, out_size,
				AudioTrack.MODE_STREAM);
	}
}
布局文件:

main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentTop="true"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="24dp"
        android:text="测试时间(小时/H)"
        android:textAppearance="?android:attr/textAppearanceLarge"
        android:textSize="12pt" />

    <EditText
        android:id="@+id/timelength"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_below="@+id/textView1"
        android:layout_marginTop="20dp"
        <!-- 未输入值时会显示的内容,作为输入提示-->
        android:hint="请输入测试小时数!"
        android:layout_gravity="center_horizontal"
        android:textSize="12pt"
        <!-- 限定输入值为整数 -->
        android:inputType="number"
        android:singleLine="true" />

    <Button
        android:id="@+id/playbutton"
        android:layout_width="100dip"
        android:layout_height="60dip"
        android:layout_below="@+id/timelength"
        android:layout_marginTop="120dp"
        android:onClick="onClick"
        android:text="@string/playbutton"
        android:textSize="12pt" />

    <Button
        android:id="@+id/stopbutton"
        android:layout_width="100dip"
        android:layout_height="60dip"
        android:layout_alignBaseline="@+id/playbutton"
        android:layout_alignBottom="@+id/playbutton"
        android:layout_alignParentRight="true"
        android:onClick="onClick"
        android:text="@string/stopbutton"
        android:textSize="12pt" />

    <Chronometer
        android:id="@+id/chronometer"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_above="@+id/stopbutton"
        android:layout_centerHorizontal="true"
        android:layout_marginBottom="28dp"
        android:format="%s"
        android:textColor="#00FF00"
        android:textSize="80px" />

</RelativeLayout>
AndroidManifest.xml

注意添加相关权限

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.cfzz.amt"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk android:minSdkVersion="15" />
    <span style="background-color: rgb(102, 255, 255);"><uses-permission android:name="android.permission.RECORD_AUDIO" /></span>
    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <activity android:name="org.achartengine.GraphicalActivity" />
        <activity
            android:name="com.cfzz.amt.AndroidMikeTestActivity"
            android:screenOrientation="portrait"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

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

</manifest>
运行效果图:



        到此,麦克风测试程序就可以运行了,在编辑框内输入小时数,再点击开始即可运行麦克风,这时可以从听筒中听到麦克风里的声音,当时间到达设定时间时会自动停止,同时计时器也会暂停。当按停止按钮时,时间设置和计时器会同时置为00:00,以便新的测试。



版权声明:本文为博主原创文章,转载请注明出处。 举报

相关文章推荐

安卓手机电池信息的获取与显示

我们在开发某些安卓应用时需要获取电池的一些状态信息如:电量、电压、温度等,本文实现了电池信息的实时获取与显示。

Android 录音实现方法、仿微信语音、麦克风录音、发送语音

效果图(注!由于使用的模拟器录制,所以图片中的录音时候话筒上下波动比较小,手机上正常!): 使用方法: 录音工具类:AudioRecoderUtils.java,代码...

程序员升职加薪指南!还缺一个“证”!

CSDN出品,立即查看!

Android 实时获取麦克风输入音量的代码

Android 上有一些很有趣的应用,例如《吹裙子》、《吹气球》之类的。利用的是实时获取麦克风输入音量,然后进行相应的处理。录音过程中怎样获得声音的大小呢?网上也不少人问如何处理这个事情,也有一些解答...

Android中实现听筒中播放声音

近期有不少客户反馈到AnyChat SDK for Android能否实现从听筒中播放声音,AnyChat团队经过研究后,找到了解决方案,在现有的SDK基础上面做如下两点修改: 一、修改AnyCha...

Android话筒音频采样和使用

Android手机话筒采样有部分代码是在网上找的,具体不记得了。 使用采样结果以及是自己的app的功能代码: 1. 在一部分app中会需要对话筒的音频输入获取使用,比如之前很火的吹裙子; 2. 采样线...

android中如何控制声音是从扬声器还是从 听筒里出啦

我的一个一边拍照一边录音的程序,在语音的时候,导致我拍照的声音本是从扬声器中出来的,结果变成了声音从听筒中出来,是什么原因?求高手指教!!!

Android Receiver(听筒模式和扬声器模式,铃音模式)

切换为听筒模式声音由听筒发出 am = (AudioManager)getSystemService(Context.AUDIO_SERVICE);//听筒模式 切换为正常模式 am....

安卓手机皮套功能的开启

皮套功能说白了就是在装有翻盖带视窗皮套的手机上,在不翻开皮套盖时,通过在视窗上显示的界面直接进行一些点触操作,可直接查看或打开手机的一些功能,比如说时间日历、通话记录、短信、音乐等。这些功能的实现首先...

安卓计时器开发

我们在开发安卓应用时,经常会用到计时器功能,用来设定一些计时任务或记录某一状态的维持时间。在安卓的SDK中有专门的计时器控件(android.widget.Chronometer),我们只需在源码中添...

安卓照相机开发

目前的智能手机大都已支持照相机应用,在安卓开发中也经常会用到照相机这一模块,那么该如何调用这一模块呢?一般有两种方法,不过也是根据具体需要来定,如果只是调用照相机应用,不涉及其他操作,建议直接调用系统...
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)