深入探讨 Android 传感器

fmms 12年前
     <p>Android 是一个面向应用程序开发的富平台,它拥有许多具有吸引力的用户界面元素和数据管理功能。Android 还提供了一组丰富的接口选项。在本文中,学习如何配合使用 Android 的各种传感器选项监控您的环境。样例代码展示了如何在 Android 电话中录制音频。想构建自己的婴儿监视器吗?想用声音来接听电话或者打开房门吗?请学习如何利用配备有 Android 的设备的硬件功能。</p>    <p>对于 Java™ 开发人员来说,Android 平台是通过使用硬件传感器创建创新应用程序的理想平台。我们将学习一些可用于 Android 应用程序的接口连接选项,包括使用传感器子系统和录制音频片段。</p>    <p>利 用配备 Android 的设备的硬件功能可以构建哪些应用程序呢?任何需要电子监视和监听的应用程序都可以构建。婴儿监视器、安全系统,甚至地震仪都可以。理论上讲,您不能同时 出现在两个地方,但 Android 可以利用一些可行的方法实现这一点。纵观本文始末,您必须记住,使用的 Android 设备不仅仅局限于 “手机”,还可以是部署在固定位置、具有无线网络连接的设备,比如 EDGE 或 WiFi。</p>    <p>使用 Android 平台有一个很新颖的地方,那就是您可以在设备内部访问一些 “好工具”。过去,访问设备底层硬件的能力一度让移动开发人员感到非常棘手。尽管 Android Java 环境的角色仍然是您和设备的桥梁,但 Android 开发团队让许多硬件功能浮出了水面。该平台是一个开源平台,因此您可以自由地编写代码实现您的任务。</p>    <p>如果尚未安装 Android,您可以 下载 Android SDK。您还可以 浏览 android.hardware 包的内容并参考本文的示例。android.media 包 包含了一些提供有用和新颖功能的类。</p>    <p>Android SDK 中包含的一些面向硬件的功能描述如下。</p>    <p><br /> <a name="table1" rel="nofollow"><strong>表 1. Android SDK 中提供的面向硬件的特性</strong></a></p>    <table class="ke-zeroborder" border="0" cellspacing="0" cellpadding="0">     <tbody>      <tr>       <th>特性</th>       <th>描述</th>      </tr>      <tr>       <td><code>android.hardware.Camera</code></td>       <td>允许应用程序与相机交互的类,可以截取照片、获取预览屏幕的图像,修改用来治理相机操作的参数。</td>      </tr>      <tr>       <td><code>android.hardware.SensorManager</code></td>       <td>允许访问 Android 平台传感器的类。并非所有配备 Android 的设备都支持 <code>SensorManager</code> 中的所有传感器,虽然这种可能性让人非常兴奋。(可用传感器的简介见下文)</td>      </tr>      <tr>       <td><code>android.hardware.SensorListener</code></td>       <td>在传感器值实时更改时,希望接收更新的类要实现的接口。应用程序实现该接口来监视硬件中一个或多个可用传感器。例如,本文中的 代码 包含实现该接口的类,实现后可以监视设备的方向和内置的加速表。</td>      </tr>      <tr>       <td><code>android.media.MediaRecorder</code></td>       <td>用于录制媒体样例的类,对于录制特定位置(比如婴儿保育)的音频活动非常有用。还可以分析音频片段以便在访问控件或安全应用程序时进行身份鉴定。例如,它可以帮助您通过声音打开门,以节省时间,不需要从房产经纪人处获取钥匙。</td>      </tr>      <tr>       <td><code>android.FaceDetector</code></td>       <td>允许对人脸(以位图形式包含)进行基本识别的类。不可能有两张完全一样的脸。可以使用该类作为设备锁定方法,无需记密码 — 这是手机的生物特征识别功能。</td>      </tr>      <tr>       <td>android.os.*</td>       <td>包含几个有用类的包,可以与操作环境交互,包括电源管理、文件查看器、处理器和消息类。和许多可移动设备一样,支持 Android 的电话可能会消耗大量电能。让设备在正确的时间 “醒来” 以监视感兴趣的事件是在设计时需要首先关注的方面。</td>      </tr>      <tr>       <td><code>java.util.Date</code><br /> <code>java.util.Timer</code><br /> <code>java.util.TimerTask</code></td>       <td>当测量实际的事件时,数据和时间往往很重要。例如,<code>java.util.Date</code> 类允许您在遇到特定的事件或状况时获取时间戳。您可以使用 <code>java.util.Timer</code> 和 <code>java.util.TimerTask</code> 分别执行周期性任务或时间点任务。</td>      </tr>     </tbody>    </table>    <p> </p>    <p>android.hardware.SensorManager 包含几个常量,这表示 Android 传感器系统的不同方面,包括:</p>    <dl>     <dt>      <strong>传感器类型</strong>     </dt>     <dd>      方向、加速表、光线、磁场、临近性、温度等。     </dd>     <dt>      <strong>采样率</strong>     </dt>     <dd>      最快、游戏、普通、用户界面。当应用程序请求特定的采样率时,其实只是对传感器子系统的一个提示,或者一个建议。不保证特定的采样率可用。     </dd>     <dt>      <strong>准确性</strong>     </dt>     <dd>      高、低、中、不可靠。     </dd>    </dl>    <p><code>SensorListener</code> 接口是传感器应用程序的中心。它包括两个必需方法:</p>    <ul>     <li><code>onSensorChanged(int sensor,float values[])</code> 方法在传感器值更改时调用。该方法只对受此应用程序监视的传感器调用(更多内容见下文)。该方法的参数包括:一个整数,指示更改的传感器;一个浮点值数组,表示传感器数据本身。有些传感器只提供一个数据值,另一些则提供三个浮点值。方向和加速表传感器都提供三个数据值。</li>     <li>当传感器的准确性更改时,将调用 <code>onAccuracyChanged(int sensor,int accuracy)</code> 方法。参数包括两个整数:一个表示传感器,另一个表示该传感器新的准确值。</li>    </ul>    <p>要与传感器交互,应用程序必须注册以侦听与一个或多个传感器相关的活动。注册使用 <code>SensorManager</code> 类的 <code>registerListener</code> 方法完成。本文中的 代码示例 演示了如何注册和注销 <code>SensorListener</code>。</p>    <p>记住,并非所有支持 Android 的设备都支持 SDK 中定义的所有传感器。如果某个传感器无法在特定的设备上使用,您的应用程序就会适当地降级。</p>    <p>样例应用程序仅监控对方向和加速表传感器的更改(源代码见 下载)。当收到更改时,传感器值在 <code>TextView</code> 小部件的屏幕上显示。图 1 展示了该应用程序的运行情况。</p>    <p><br /> <a name="N10173" rel="nofollow"><strong>图 1. 监视加速和方向</strong></a><br /> <img alt="深入探讨 Android 传感器" src="https://simg.open-open.com/show/392434394f465f6a73d96b755a10289b.jpg" width="320" height="480" /> <br /> </p>    <p>使用 Eclipse 环境和 Android Developer Tools 插件创建的应用程序。(关于使用 Eclipse 开发 Android 应用程序的信息,请参见 参考资料。)清单 1 展示了该应用程序的代码。</p>    <p><br /> <a name="list1" rel="nofollow"><strong>清单 1. IBMEyes.java</strong></a></p>    <pre class="brush:java; toolbar: true; auto-links: false;">package com.msi.ibm.eyes; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.widget.TextView; import android.hardware.SensorManager; import android.hardware.SensorListener; public class IBMEyes extends Activity implements SensorListener {     final String tag = "IBMEyes";     SensorManager sm = null;     TextView xViewA = null;     TextView yViewA = null;     TextView zViewA = null;     TextView xViewO = null;     TextView yViewO = null;     TextView zViewO = null;      /** Called when the activity is first created. */     @Override     public void onCreate(Bundle savedInstanceState) {         super.onCreate(savedInstanceState);        // get reference to SensorManager         sm = (SensorManager) getSystemService(SENSOR_SERVICE);         setContentView(R.layout.main);         xViewA = (TextView) findViewById(R.id.xbox);         yViewA = (TextView) findViewById(R.id.ybox);         zViewA = (TextView) findViewById(R.id.zbox);         xViewO = (TextView) findViewById(R.id.xboxo);         yViewO = (TextView) findViewById(R.id.yboxo);         zViewO = (TextView) findViewById(R.id.zboxo);     }     public void onSensorChanged(int sensor, float[] values) {         synchronized (this) {             Log.d(tag, "onSensorChanged: " + sensor + ", x: " +  values[0] + ", y: " + values[1] + ", z: " + values[2]);             if (sensor == SensorManager.SENSOR_ORIENTATION) {                 xViewO.setText("Orientation X: " + values[0]);                 yViewO.setText("Orientation Y: " + values[1]);                 zViewO.setText("Orientation Z: " + values[2]);             }             if (sensor == SensorManager.SENSOR_ACCELEROMETER) {                 xViewA.setText("Accel X: " + values[0]);                 yViewA.setText("Accel Y: " + values[1]);                 zViewA.setText("Accel Z: " + values[2]);             }                     }     }          public void onAccuracyChanged(int sensor, int accuracy) {      Log.d(tag,"onAccuracyChanged: " + sensor + ", accuracy: " + accuracy);     }     @Override     protected void onResume() {         super.onResume();       // register this class as a listener for the orientation and accelerometer sensors         sm.registerListener(this,                  SensorManager.SENSOR_ORIENTATION |SensorManager.SENSOR_ACCELEROMETER,                 SensorManager.SENSOR_DELAY_NORMAL);     }          @Override     protected void onStop() {         // unregister listener         sm.unregisterListener(this);         super.onStop();     }     }</pre>    <p>编写应用程序必须基于常见的活动,因为它只是利用从传感器获取的数据更新屏幕。在设备可能在前台执行其他活动的应用程序中,将应用程序构建为服务可能更加合适。</p>    <p>该活动的 <code>onCreate</code> 方法可以引用 <code>SensorManager</code>,其中包含所有与传感器有关的函数。<code>onCreate</code> 方法还建立了对 6 个 <code>TextView</code> 小部件的引用,您需要使用传感器数据值更新这些小部件。</p>    <p><code>onResume()</code> 方法使用对 <code>SensorManager</code> 的引用通过 <code>registerListener</code> 方法注册传感器更新:</p>    <ul>     <li>第一个参数是实现 <code>SensorListener</code> 接口的类的实例。 </li>     <li>第二个参数是所需传感器的位掩码。在本例中,应用程序从 <code>SENSOR_ORIENTATION</code> 和 <code>SENSOR_ACCELEROMETER</code> 请求数据。</li>     <li>第三个参数是一个系统提示,指出应用程序更新传感器值所需的速度。</li>    </ul>    <p>应用程序(活动)暂停后,需要注销侦听器,这样以后就不会再收到传感器更新。这通过 <code>SensorManager</code> 的 <code>unregisterListener</code> 方法实现。惟一的参数是 <code>SensorListener</code> 的实例。</p>    <p>在 <code>registerListener</code> 和 <code>unregisterListener</code> 方法调用中,应用程序使用关键字 <code>this</code>。注意类定义中的 <code>implements</code> 关键字,其中声明了该类实现 <code>SensorListener</code> 接口。这就是要将它传递到 <code>registerListener</code> 和 <code>unregisterListener</code> 的原因。</p>    <p><code>SensorListener</code> 必须实现两个方法 <code>onSensorChange</code> 和 <code>onAccuracyChanged</code>。示例应用程序不关心传感器的准确度,但关注传感器当前的 X、Y 和 Z 值。<code>onAccuracyChanged</code> 方法实质上不执行任何操作;它只在每次调用时添加一个日志项。</p>    <p>似乎经常需要调用 <code>onSensorChanged</code> 方法,因为加速表和方向传感器正在快速发送数据。查看第一个参数确定哪个传感器在发送数据。确认了发送数据的传感器之后,将使用方法第二个参数传递的浮点 值数组中所包含的数据更新相应的 UI 元素。该示例只是显示这些值,但在更加高级的应用程序中,还可以分析这些值,比较原来的值,或者设置某种模式识别算法来确定用户(或外部环境)的行为。</p>    <p>现在您已经了解了传感器子系统,接下来的部分将回顾一个在 Android 手机上录制音频的代码样例。该样例运行在 DEV1 开发设备上。</p>    <p><a name="N10218" rel="nofollow"><span>使用 MediaRecorder</span></a></p>    <p>android.media 包包含与媒体子系统交互的类。使用 <code>android.media.MediaRecorder</code> 类进行媒体采样,包括音频和视频。<code>MediaRecorder</code> 作为状态机运行。您需要设置不同的参数,比如源设备和格式。设置后,可执行任何时间长度的录制,直到用户停止。</p>    <p>清单 2 包含的代码在 Android 设备上录制音频。显示的代码不包括应用程序的 UI 元素(完整源代码见 下载)。</p>    <p><br /> <a name="N10234" rel="nofollow"><strong>清单 2. 录制音频片段</strong></a></p>    <pre class="brush:java; toolbar: true; auto-links: false;">MediaRecorder mrec ; File audiofile = null; private static final String TAG="SoundRecordingDemo"; protected void startRecording() throws IOException  {    mrec.setAudioSource(MediaRecorder.AudioSource.MIC);    mrec.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);    mrec.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);    if (mSampleFile == null)     {        File sampleDir = Environment.getExternalStorageDirectory();        try         {            audiofile = File.createTempFile("ibm", ".3gp", sampleDir);        }        catch (IOException e)         {            Log.e(TAG,"sdcard access error");            return;        }    }    mrec.setOutputFile(audiofile.getAbsolutePath());    mrec.prepare();    mrec.start(); } protected void stopRecording()  {    mrec.stop();    mrec.release();    processaudiofile(audiofile.getAbsolutePath()); } protected void processaudiofile()  {    ContentValues values = new ContentValues(3);    long current = System.currentTimeMillis();    values.put(MediaStore.Audio.Media.TITLE, "audio" + audiofile.getName());    values.put(MediaStore.Audio.Media.DATE_ADDED, (int) (current / 1000));    values.put(MediaStore.Audio.Media.MIME_TYPE, "audio/3gpp");    values.put(MediaStore.Audio.Media.DATA, audiofile.getAbsolutePath());    ContentResolver contentResolver = getContentResolver();        Uri base = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;    Uri newUri = contentResolver.insert(base, values);        sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, newUri)); }</pre>    <p>在 <code>startRecording</code> 方法中,实例化并初始化 <code>MediaRecorder</code> 的实例:</p>    <ul>     <li>输入源被设置为麦克风(<code>MIC</code>)。</li>     <li>输出格式被设置为 3GPP(*.3gp 文件),这是移动设备专用的媒体格式。</li>     <li>编码器被设置为 <code>AMR_NB</code>,这是音频格式,采样率为 8 KHz。NB 表示窄频。SDK 文档 解释了不同的数据格式和可用的编码器。</li>    </ul>    <p>音频文件存储在存储卡而不是内存中。<code>External.getExternalStorageDirectory()</code> 返回存储卡位置的名称,在该目录中将创建一个临时文件名。然后,通过调用 <code>setOutputFile</code> 方法将文件关联到 <code>MediaRecorder</code> 实例。音频数据将存储到该文件中。</p>    <p>调用 <code>prepare</code> 方法完成 <code>MediaRecorder</code> 的初始化。准备开始录制流程时,将调用 <code>start</code> 方法。在调用 <code>stop</code> 方法之前,将对存储卡上的文件进行录制。release 方法将释放分配给 <code>MediaRecorder</code> 实例的资源。</p>    <p>音频采样完成之后,需要采取以下步骤:</p>    <ul>     <li>向设备的媒体库添加该音频。</li>     <li>执行一些模式识别步骤确定声音:      <ul>       <li>这是婴儿的啼哭声吗?</li>       <li>这是所有人的声音吗?是否要解锁手机?</li>       <li>这是 “芝麻开门” 吗?是否要打开通往 “秘密通道” 的大门?</li>      </ul> </li>     <li>自动将音频文件上传到网络位置以便处理。</li>    </ul>    <p>在该代码样例中,<code>processaudiofile</code> 方法将音频添加到媒体库。使用 <code>Intent</code> 通知设备上的媒体应用程序有新内容可用。</p>    <p>关于该代码片段最后要注意的是:如果您试用,它一开始不会录制音频。您将看到创建的文件,但是没有任何音频。您需要向 AndroidManifest.xml 文件添加权限:</p>    <pre class="brush:xml; toolbar: true; auto-links: false;"><uses-permission android:name="android.permission.RECORD_AUDIO"></uses-permission></pre>    <br /> 现在,您已经学了一点关于与 Android 传感器和录制音频相关的内容。下一节将更全面的介绍与数据采集和报告系统有关的应用程序架构。    <p></p>    <p><a name="N102B0" rel="nofollow"><span>Android 作为传感器平台</span></a></p>    <p>Android 平台包含各种用于监视环境的传感器选项。有了输入或模拟选项数组,以及高级计算和互联功能,Android 成为构建实际系统的最佳平台。图 2 显示了输入、应用程序逻辑、通知方法或输出之间的简单视图。</p>    <p><br /> <a name="N102BB" rel="nofollow"><strong>图 2. 以 Android 为中心的传感器系统的方块图</strong></a><br /> <img alt="深入探讨 Android 传感器" src="https://simg.open-open.com/show/f2d6035b5aa2ec92f8ffd2eb1353e40a.gif" width="351" height="153" /> <br /> </p>    <p>该架构很灵活;应用程序逻辑可以划分为本地 Android 设备和服务器端资源(可以实现更大的数据库和计算功能)。例如,本地 Android 设备上录制的音轨可以 <code>POST</code> 到 Web 服务器,其中将根据音频模式数据库比较数据。很明显,这仅仅是冰山一角。希望您能更深入地研究,让 Android 平台超越移动电话的范畴。</p>    <p><a name="N102D0" rel="nofollow"><span>结束语</span></a></p>    <p>在本文中,我们介绍了 Android 传感器。样例应用程序度量了方向和加速,以及使用 <code>MediaRecorder</code> 类与录制功能进行交互。对于构建实际系统,Android 是一个灵活、有吸引力的平台。Android 领域发展迅速,并且不断壮大。请务必关注该平台。</p>    <p><span><a name="download" rel="nofollow">下载</a></span></p>    <table class="ke-zeroborder" border="0" cellspacing="0" cellpadding="0">     <tbody>      <tr>       <td>描述</td>       <td>名字</td>       <td>大小</td>       <td>下载方法</td>      </tr>      <tr>       <td>Eyes 源代码</td>       <td>os-android-sensorEyes.zip</td>       <td>28KB</td>       <td><a href="/misc/goto?guid=4959500234788565647" rel="nofollow"><strong>HTTP</strong></a></td>      </tr>      <tr>       <th>IBMAudio 源代码</th>       <td>os-android-sensorIBMAudio.zip</td>       <td>33KB</td>       <td><a href="/misc/goto?guid=4959500234875806698" rel="nofollow"><strong>HTTP</strong></a></td>      </tr>     </tbody>    </table>