android - 如何在手机空闲状态下采集加速度计数据

标签 android accelerometer android-sensors

我目前正在开发一个应用程序,该应用程序应该每两秒收集一次加速度计和麦克风数据。为此,我启动了一项服务(在内部运行一个计时器),以便在应用程序在后台运行时收集数据。到目前为止一切顺利。

计时器启动、收集数据并将该数据写入 XML 文件。但在查看生成的 XML 文件后,我意识到,当手机处于空闲状态时,加速度计值始终相同。我什至使用唤醒锁来每两秒“唤醒”一次手机,但即使那样也不起作用。

相关代码如下:

    @Override
public void onCreate() {
    super.onCreate();

    Toast.makeText(this, "Background Service Created", Toast.LENGTH_SHORT)
            .show();
    Log.d(TAG, "onCreate Service");

    this.startActivities();
}

@Override
public void onDestroy() {
    super.onDestroy();

    // Kill timer, sensors and XML file
    this.stopActivities();

    Toast.makeText(this, "Background Service Stopped", Toast.LENGTH_SHORT)
            .show();
    Log.d(TAG, "onDestroy");
}

@Override
public void onStart(Intent intent, int startid) {
    Toast.makeText(this, "Background Service Started", Toast.LENGTH_SHORT).show();
    Log.d(TAG, "onStart");
}

/**
 * Expanded modification function. 
 */
private void startExpandedNotification(){
    //Id for the intent of the expanded notification
    final int myID = 1234;

    //The intent to launch when the user clicks the expanded notification
    Intent i = new Intent(this, MainMovement.class);
    i.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
    PendingIntent pendIntent = PendingIntent.getActivity(this, 0, i, 0);

    //This constructor is deprecated. Use Notification.Builder instead if programming for 3.0+
    Notification notice = new Notification(R.drawable.ic_launcher, "Movement Sampling Activated", System.currentTimeMillis());

    //This method is deprecated. Use Notification.Builder instead if programming for 3.0+
    notice.setLatestEventInfo(this, "Movement Sampling", "Sensors activated", pendIntent);

    notice.flags |= Notification.FLAG_NO_CLEAR;
    startForeground(myID, notice);
}

/**
 * Stop Expanded Notification
 */
private void stopExpandedNotification(){
    stopForeground(true);
}

/**
 * Starts everything that the service needs to collect the sensors data 
 * (timer, xml file, sensors, expanded notification)
 */
private void startActivities() {
    // Define output file path
    OUTPUT_FILE = Environment.getExternalStorageDirectory()
            .getAbsolutePath() + "/Ze";// also added ""/Ze here
    OUTPUT_FILE += "/audiorecordtest.3gp";

    timer = new Timer(); // If I take this line out, it will lead to an
                            // exception. But if I do so, the acc values are
                            // measured while the phone is in the idle state (works only on samsung spika)
    timer.scheduleAtFixedRate(new mainTask(), 0, _refreshRate);

    xml = new DataStore();

    this.startRecordingMic();
    this.startRecordingAcc();

    startExpandedNotification();

    _sampling = true;
}

/**
 * Stops everything that the service needs to collect the sensors data 
 * (timer, xml file, sensors, expanded notification)
 */
private void stopActivities() {
    timer.cancel();
    //Log.e(TAG, "timer: CANCELLED");

    // close xml file
    xml.closeFile();
    //Log.e(TAG, "XML: CLOSED");

    // unregister sensor
    _sensorManager.unregisterListener(accelerationListener);
    //Log.e(TAG, "Sensors: UNREGISTERED");

    // stop recording
    if (_recorder != null) {
        _recorder.stop();
        _recorder.release();
        _recorder = null;
    }

    stopExpandedNotification();

    _sampling = false;
}

/**
 * Responsible for collecting the sensors data every two seconds
 */
private class mainTask extends TimerTask {
    /**
     * Collects the sensor data every 2 seconds
     */
    @Override
    public void run() {
        MovementService.this.collectData();
    }
}

/**
 * Initiates the microphone in order to collect the amplitude value
 */
private void startRecordingMic() {
    _recorder = new MediaRecorder();
    _recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
    _recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
    _recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
    _recorder.setOutputFile(OUTPUT_FILE);

    try {
        _recorder.prepare();
    } catch (final IllegalStateException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (final IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }

    _recorder.start();
}

/**
 * Initiates the acceleromenter sensor in order to collect movement data
 */
private void startRecordingAcc() {
    // ACCELEROMETER
    _sensorManager = (SensorManager) this
            .getSystemService(Context.SENSOR_SERVICE);
    _accelerometer = _sensorManager
            .getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
    _sensorManager.registerListener(accelerationListener, _accelerometer,
            SensorManager.SENSOR_DELAY_FASTEST);
}

private final SensorEventListener accelerationListener = new SensorEventListener() {

    public void onAccuracyChanged(Sensor sensor, int acc) {
    }


    public void onSensorChanged(SensorEvent event) {

        // sampling values per axis
        _x = event.values[0];
        _y = event.values[1];
        _z = event.values[2];

        //Log.e(TAG , "x: " + _x + "  y: " + _y + " z:" + _z);
    }
};

/**
 * Send data (amplitude and decibel) to the xml file
 */
public void collectData() {
    //The power lock ensures that the service will indeed collect the accelerometer data
    PowerManager pm = (PowerManager) MovementService.this.getSystemService(Context.POWER_SERVICE);
    PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.FULL_WAKE_LOCK | PowerManager.ACQUIRE_CAUSES_WAKEUP, "YOUR TAG");
    wl.acquire();
    Log.v(TAG, "lock aquired");

    if (xml != null) xml.writeOnFile(this.getAmplitude(), this.getAccelaration(), this.getTotalAccelaration());

    wl.release();
    Log.v(TAG, "lock released");
}

你们有什么解决这个问题的建议吗?这让我发疯。

Ps:我用的是三星spika,安卓2.1版本。

最佳答案

据我所知,设备空闲时无法对加速度计进行采样。您需要设置一个 PARTIAL_WAKE_LOCK 以保持 CPU 运行。它仍然允许屏幕关闭,但当然,电池会比正常情况更快地耗尽。

http://developer.android.com/reference/android/os/PowerManager.html#PARTIAL_WAKE_LOCK

代码:

PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "My Tag");
wl.acquire();
//  ..screen will stay on during this section..
wl.release();

您还必须在 list 中包含此权限条目:

<uses-permission android:name="android.permission.WAKE_LOCK" /> 

关于这一点之前有一个问题: How to sample accelerometer data in the background or on the sleep mode

关于android - 如何在手机空闲状态下采集加速度计数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9485172/

相关文章:

android - 如何在 osmdroid 中设置最大缩放?

android - 如何在android中对齐布局中的文本

ios - 当方向锁定打开且应用程序仅支持纵向时,如何检测设备方向

android - VideoView 播放状态或 MediaController 播放/暂停的事件

java - 如何让信任 anchor 在 Android API 级别 16-19 上正常工作

java - 计算纵跳腾空时间

android - 指南针在 android 中不起作用

java - 静止不动时使用 TYPE_ROTATION_VECTOR 实现当前基本方向方法?

android - Android 手机中的加速度计和陀螺仪及其功耗

android - 使用地理定位 api cordova/phonegap 查找距离、速度和加速度