Android���用于传感器的后台服务

3
我已经编写了一段代码,可以在我们晃动设备时打开一个应用程序。只要我将所有的代码放到MainActivity中,而且没有任何后台服务,一切都很顺利。但是当我尝试添加后台服务时,它就无法正常工作了,因为我希望它能够持续运行而不需要每次都进入应用程序。所以有人能告诉我我错在哪里吗?谢谢!
BackgroundService.java
public class BackgroundService extends Service implements SensorEventListener {

    public BackgroundService() {
    }
    private SensorManager mSensorManager;
    private Sensor mSensor;
    private long lastUpdate = 0;
    private float last_x, last_y, last_z;
    private static final int SHAKE_THRESHOLD = 600;


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

        // Get sensor manager on starting the service.
        mSensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);

        // Registering...
        mSensorManager.registerListener(this, mSensor, SensorManager.SENSOR_DELAY_NORMAL);

        // Get default sensor type
        mSensor = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {

        // Get sensor manager on starting the service.
        mSensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);

        // Registering...
        mSensorManager.registerListener(this, mSensor, SensorManager.SENSOR_DELAY_NORMAL);

        // Get default sensor type
        mSensor = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);

        return START_STICKY;

    }

    @Override
    public IBinder onBind(Intent intent) {
        throw new UnsupportedOperationException("Not Yet Implemented");
    }

    @Override
    public void onSensorChanged(SensorEvent sensorEvent) {

        mSensorManager.registerListener(this, mSensor, SensorManager.SENSOR_DELAY_NORMAL);

        Sensor mySensor = sensorEvent.sensor;

        mSensorManager.registerListener(this, mySensor, SensorManager.SENSOR_DELAY_NORMAL);

        if (mySensor.getType() == Sensor.TYPE_ACCELEROMETER) {

            float[] values = sensorEvent.values;
            float x = values[0];
            float y = values[1];
            float z = values[2];

            long curTime = System.currentTimeMillis();
            if ((curTime - lastUpdate) > 100) {
                long diffTime = (curTime - lastUpdate);
                lastUpdate = curTime;
                float speed
                        = Math.abs(x + y + z - last_x - last_y - last_z) / diffTime * 10000;
                if (speed > SHAKE_THRESHOLD) {
                    openWhatsApp();
                }
                last_x = x;
                last_y = y;
                last_z = z;
            }
            // Stop the sensor and service
               mSensorManager.unregisterListener(this);
               stopSelf();
        }
    }

    @Override
    public void onAccuracyChanged(Sensor sensor, int i) {
    }


    public void openWhatsApp() {
        Intent launchIntent = getPackageManager().getLaunchIntentForPackage("com.whatsapp");
        if (launchIntent != null) {
            startService(launchIntent);
        }
    }

}

MainActivity.java

公共类 MainActivity 扩展自 AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        Intent i = new Intent(MainActivity.this, BackgroundService.class);
        startService(i);
    }
}

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="joshipinak.com.shakeshooksaken">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name"
            android:theme="@style/AppTheme.NoActionBar">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

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

        </activity>
        <service android:name=".BackgroundService" />
    </application>

</manifest>

欢迎来到Stackoverflow。这真的是一个最小化的示例吗? - loki
下次会记住的。@loki :) - Pinak Joshi
你可以使用rxJava代替。 - ste9206
1个回答

5
GitHub Android Examples中有一个完整的应用程序,它使用命令模式来检查传感器值和AlarmManager,与您的代码类似。
这个关于SO的问题中提供了一个使用SensorEventListener接口的示例。
Android开发者网站在此页面上建议使用IntentService。 您还可以在这个SO问题中阅读Service和IntentService之间的区别。
这些链接也可能对您有帮助:

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接