上次简单地介绍了AudioRecord和AudioTrack的使用,这次就结合SurfaceView实现一个Android版的手机模拟信号示波器。最近物联网炒得很火,作为手机软件开发者,如何在不修改手机硬件电路的前提下实现与第三方传感器结合呢?麦克风就是一个很好的ADC接口,通过麦克风与第三方传感器结合,再在软件里对模拟信号做相应的处理,就可以提供更丰富的传感化应用。
       先来看看本文程序运行的效果图(屏幕录像速度较慢,真机实际运行起来会更加流畅):

       本文程序使用8000hz的采样率,对X轴方向绘图的实时性要求较高,如果不降低X轴的分辨率,程序的实时性较差,因此程序对X轴数据缩小区间为8倍~16倍。由于采用16位采样,因此Y轴数据的高度相对于手机屏幕来说也偏大,程序也对Y轴数据做缩小,区间为1倍~10倍。在SurfaceView的OnTouchListener方法里加入了波形基线的位置调节,直接在SurfaceView控件上触摸即可控制整体波形偏上或偏下显示。
       main.xml源码如下:
XML/HTML代码
    - <linearlayout xmlns:android="http://schemas.android.com/apk/res/android"   
-         android:orientation="vertical" android:layout_width="fill_parent"  
-         android:layout_height="fill_parent">  
-         <linearlayout android:id="@+id/LinearLayout01"   
-                 android:layout_height="wrap_content" android:layout_width="fill_parent"  
-                 android:orientation="horizontal">  
-                 <button android:layout_height="wrap_content" android:id="@+id/btnStart"   
-                         android:text="开始" android:layout_width="80dip">  
-                 <button android:layout_height="wrap_content" android:text="停止"   
-                         android:id="@+id/btnExit" android:layout_width="80dip">  
-                 <zoomcontrols android:layout_width="wrap_content"   
-                         android:layout_height="wrap_content" android:id="@+id/zctlX">  
-                 <zoomcontrols android:layout_width="wrap_content"   
-                         android:layout_height="wrap_content" android:id="@+id/zctlY">  
-           
-         <surfaceview android:id="@+id/SurfaceView01"   
-                 android:layout_height="fill_parent" android:layout_width="fill_parent">  
       ClsOscilloscope.java是实现示波器的类库,包含AudioRecord操作线程和SurfaceView绘图线程的实现,两个线程同步操作,代码如下:
Java代码
    - package com.testOscilloscope;  
- import java.util.ArrayList;  
- import android.graphics.Canvas;  
- import android.graphics.Color;  
- import android.graphics.Paint;  
- import android.graphics.Rect;  
- import android.media.AudioRecord;  
- import android.view.SurfaceView;  
- public class ClsOscilloscope {  
-         private ArrayList inBuf = new ArrayList();  
-         private boolean isRecording = false;  
-          
-  
-   
-         public int rateX = 4;  
-          
-  
-   
-         public int rateY = 4;  
-          
-  
-   
-         public int baseLine = 0;  
-          
-  
-   
-         public void initOscilloscope(int rateX, int rateY, int baseLine) {  
-                 this.rateX = rateX;  
-                 this.rateY = rateY;  
-                 this.baseLine = baseLine;  
-         }  
-          
-  
-  
-  
-  
-   
-         public void Start(AudioRecord audioRecord, int recBufSize, SurfaceView sfv,  
-                         Paint mPaint) {  
-                 isRecording = true;  
-                 new RecordThread(audioRecord, recBufSize).start();  
-                 new DrawThread(sfv, mPaint).start();  
-         }  
-          
-  
-   
-         public void Stop() {  
-                 isRecording = false;  
-                 inBuf.clear();  
-         }  
-          
-  
-  
-  
-  
-   
-         class RecordThread extends Thread {  
-                 private int recBufSize;  
-                 private AudioRecord audioRecord;  
-                 public RecordThread(AudioRecord audioRecord, int recBufSize) {  
-                         this.audioRecord = audioRecord;  
-                         this.recBufSize = recBufSize;  
-                 }  
-                 public void run() {  
-                         try {  
-                                 short[] buffer = new short[recBufSize];  
-                                 audioRecord.startRecording();  
-                                 while (isRecording) {  
-                                           
-                                         int bufferReadResult = audioRecord.read(buffer, 0,  
-                                                         recBufSize);  
-                                         short[] tmpBuf = new short[bufferReadResult / rateX];  
-                                         for (int i = 0, ii = 0; i < tmpBuf.length; i++, ii = i  
-                                                         * rateX) {  
-                                                 tmpBuf[i] = buffer[ii];  
-                                         }  
-                                         synchronized (inBuf) {  
-                                                 inBuf.add(tmpBuf);  
-                                         }  
-                                 }  
-                                 audioRecord.stop();  
-                         } catch (Throwable t) {  
-                         }  
-                 }  
-         };  
-          
-  
-  
-  
-  
-   
-         class DrawThread extends Thread {  
-                 private int oldX = 0;  
-                 private int oldY = 0;  
-                 private SurfaceView sfv;  
-                 private int X_index = 0;  
-                 private Paint mPaint;  
-                 public DrawThread(SurfaceView sfv, Paint mPaint) {  
-                         this.sfv = sfv;  
-                         this.mPaint = mPaint;  
-                 }  
-                 public void run() {  
-                         while (isRecording) {  
-                                 ArrayList buf = new ArrayList();  
-                                 synchronized (inBuf) {  
-                                         if (inBuf.size() == 0)  
-                                                 continue;  
-                                         buf = (ArrayList) inBuf.clone();  
-                                         inBuf.clear();  
-                                 }  
-                                 for (int i = 0; i < buf.size(); i++) {  
-                                         short[] tmpBuf = buf.get(i);  
-                                         SimpleDraw(X_index, tmpBuf, rateY, baseLine);  
-                                         X_index = X_index + tmpBuf.length;  
-                                         if (X_index > sfv.getWidth()) {  
-                                                 X_index = 0;  
-                                         }  
-                                 }  
-                         }  
-                 }  
-                  
-  
-  
-  
-  
-  
-  
-  
-  
-  
-  
-   
-                 void SimpleDraw(int start, short[] buffer, int rate, int baseLine) {  
-                         if (start == 0)  
-                                 oldX = 0;  
-                         Canvas canvas = sfv.getHolder().lockCanvas(  
-                                         new Rect(start, 0, start + buffer.length, sfv.getHeight()));  
-                         canvas.drawColor(Color.BLACK);  
-                         int y;  
-                         for (int i = 0; i < buffer.length; i++) {  
-                                 int x = i + start;  
-                                 y = buffer[i] / rate + baseLine;  
-                                 canvas.drawLine(oldX, oldY, x, y, mPaint);  
-                                 oldX = x;  
-                                 oldY = y;  
-                         }  
-                         sfv.getHolder().unlockCanvasAndPost(canvas);  
-                 }  
-         }  
- }  
       testOscilloscope.java是主程序,控制UI和ClsOscilloscope,代码如下:
Java代码
    - package com.testOscilloscope;  
- import android.app.Activity;  
- import android.graphics.Color;  
- import android.graphics.Paint;  
- import android.media.AudioFormat;  
- import android.media.AudioRecord;  
- import android.media.MediaRecorder;  
- import android.os.Bundle;  
- import android.view.MotionEvent;  
- import android.view.SurfaceView;  
- import android.view.View;  
- import android.view.View.OnTouchListener;  
- import android.widget.Button;  
- import android.widget.ZoomControls;  
- public class testOscilloscope extends Activity {  
-       
-         Button btnStart,btnExit;  
-         SurfaceView sfv;  
-     ZoomControls zctlX,zctlY;  
-       
-     ClsOscilloscope clsOscilloscope=new ClsOscilloscope();  
-       
-         static final int frequency = 8000;  
-         static final int channelConfiguration = AudioFormat.CHANNEL_CONFIGURATION_MONO;  
-         static final int audioEncoding = AudioFormat.ENCODING_PCM_16BIT;  
-         static final int xMax = 16;  
-         static final int xMin = 8;  
-         static final int yMax = 10;  
-         static final int yMin = 1;  
-           
-         int recBufSize;  
-         AudioRecord audioRecord;  
-         Paint mPaint;  
-     @Override  
-     public void onCreate(Bundle savedInstanceState) {  
-         super.onCreate(savedInstanceState);  
-         setContentView(R.layout.main);  
-           
-                 recBufSize = AudioRecord.getMinBufferSize(frequency,  
-                                 channelConfiguration, audioEncoding);  
-                 audioRecord = new AudioRecord(MediaRecorder.AudioSource.MIC, frequency,  
-                                 channelConfiguration, audioEncoding, recBufSize);  
-                   
-                 btnStart = (Button) this.findViewById(R.id.btnStart);  
-                 btnStart.setOnClickListener(new ClickEvent());  
-                 btnExit = (Button) this.findViewById(R.id.btnExit);  
-                 btnExit.setOnClickListener(new ClickEvent());  
-                   
-                 sfv = (SurfaceView) this.findViewById(R.id.SurfaceView01);   
-                 sfv.setOnTouchListener(new TouchEvent());  
-         mPaint = new Paint();    
-         mPaint.setColor(Color.GREEN);  
-         mPaint.setStrokeWidth(1);  
-           
-         clsOscilloscope.initOscilloscope(xMax/2, yMax/2, sfv.getHeight()/2);  
-           
-           
-                 zctlX = (ZoomControls)this.findViewById(R.id.zctlX);  
-                 zctlX.setOnZoomInClickListener(new View.OnClickListener() {  
-                         @Override  
-                         public void onClick(View v) {  
-                                 if(clsOscilloscope.rateX>xMin)  
-                                         clsOscilloscope.rateX--;  
-                                 setTitle("X轴缩小"+String.valueOf(clsOscilloscope.rateX)+"倍"  
-                                                 +","+"Y轴缩小"+String.valueOf(clsOscilloscope.rateY)+"倍");  
-                         }  
-                 });  
-                 zctlX.setOnZoomOutClickListener(new View.OnClickListener() {  
-                         @Override  
-                         public void onClick(View v) {  
-                                 if(clsOscilloscope.rateX<xmax)  
-                                         clsOscilloscope.rateX++;          
-                                 setTitle("X轴缩小"+String.valueOf(clsOscilloscope.rateX)+"倍"  
-                                                 +","+"Y轴缩小"+String.valueOf(clsOscilloscope.rateY)+"倍");  
-                         }  
-                 });  
-                 zctlY = (ZoomControls)this.findViewById(R.id.zctlY);  
-                 zctlY.setOnZoomInClickListener(new View.OnClickListener() {  
-                         @Override  
-                         public void onClick(View v) {  
-                                 if(clsOscilloscope.rateY>yMin)  
-                                         clsOscilloscope.rateY--;  
-                                 setTitle("X轴缩小"+String.valueOf(clsOscilloscope.rateX)+"倍"  
-                                                 +","+"Y轴缩小"+String.valueOf(clsOscilloscope.rateY)+"倍");  
-                         }  
-                 });  
-                   
-                 zctlY.setOnZoomOutClickListener(new View.OnClickListener() {  
-                         @Override  
-                         public void onClick(View v) {  
-                                 if(clsOscilloscope.rateY<ymax)  
-                                         clsOscilloscope.rateY++;          
-                                 setTitle("X轴缩小"+String.valueOf(clsOscilloscope.rateX)+"倍"  
-                                                 +","+"Y轴缩小"+String.valueOf(clsOscilloscope.rateY)+"倍");  
-                         }  
-                 });  
-     }  
-         @Override  
-         protected void onDestroy() {  
-                 super.onDestroy();  
-                 android.os.Process.killProcess(android.os.Process.myPid());  
-         }  
-           
-          
-  
-  
-  
-   
-         class ClickEvent implements View.OnClickListener {  
-                 @Override  
-                 public void onClick(View v) {  
-                         if (v == btnStart) {  
-                                 clsOscilloscope.baseLine=sfv.getHeight()/2;  
-                                 clsOscilloscope.Start(audioRecord,recBufSize,sfv,mPaint);  
-                         } else if (v == btnExit) {  
-                                 clsOscilloscope.Stop();  
-                         }  
-                 }  
-         }  
-          
-  
-  
-  
-   
-         class TouchEvent implements OnTouchListener{  
-                 @Override  
-                 public boolean onTouch(View v, MotionEvent event) {  
-                         clsOscilloscope.baseLine=(int)event.getY();  
-                         return true;  
-                 }  
-                   
-         }  
- }