我有一些代码可以在Android上扩展服务并记录onSensorChanged(SensorEvent事件)加速计传感器读数。我希望能够记录这些传感器读数,即使设备关闭时(我对电池寿命非常小心,并且在运行时显而易见)。虽然屏幕上的日志工作正常,但在2.0.1 Motorola Droid和2.1 Nexus One上。
然而,当手机进入睡眠状态时(按下电源按钮)屏幕关闭,onSensorChanged
事件停止传送(每隔N次onSensorChanged
使用Log.e消息进行验证被召唤)。
该服务获取wakeLock以确保它在后台继续运行;但是,它似乎没有任何影响。我已经尝试了各种各样的PowerManager。唤醒锁,但它们似乎都不重要。
_WakeLock = _PowerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "My Tag");
_WakeLock.acquire();
关于在屏幕关闭时是否可以从传感器获取数据的报道存在冲突......任何人都对更现代版本的Android(Eclair)和硬件有任何经验吗?
这似乎表明它在Cupcake中工作: http://groups.google.com/group/android-developers/msg/a616773b12c2d9e5
PS:完全相同的代码在G1上的1.5中按预期工作。当屏幕关闭,应用程序在后台等时,记录继续进行。
答案 0 :(得分:8)
我们从2.0.1开始推断出这一点。这似乎是有意的,也许是电池寿命提升的一部分被吹捧为一个功能。我们有一个工作摇晃唤醒或解锁2.0,然后它打破了更新,我们无法得到任何解决方法。 ;'(如果保持CPU部分锁定并不总是阻止CPU休眠也没关系。从我看到的通过USB进行日志调试时,似乎偶尔会提到传感器监听器在睡眠发生时的变化。
用户发布了他声称可以在摩托罗拉设备上运行的变通方法 - https://sites.google.com/a/bug-br.org.br/android/technical-documents
我测试了解决方法,提出了教程中的以下代码和一些手动修订(他的教程中提供了一些“错误”代码):
public class ShakeWakeupService extends Service implements SensorEventListener{
private Context mContext;
SensorManager mSensorEventManager;
Sensor mSensor;
// BroadcastReceiver for handling ACTION_SCREEN_OFF.
public BroadcastReceiver mReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
// Check action just to be on the safe side.
if (intent.getAction().equals(Intent.ACTION_SCREEN_OFF)) {
Log.v("shake mediator screen off","trying re-registration");
// Unregisters the listener and registers it again.
mSensorEventManager.unregisterListener(ShakeWakeupService.this);
mSensorEventManager.registerListener(ShakeWakeupService.this, mSensor,
SensorManager.SENSOR_DELAY_NORMAL);
}
}
};
@Override
public void onCreate() {
super.onCreate();
Log.v("shake service startup","registering for shake");
mContext = getApplicationContext();
// Obtain a reference to system-wide sensor event manager.
mSensorEventManager = (SensorManager) mContext.getSystemService(Context.SENSOR_SERVICE);
// Get the default sensor for accel
mSensor = mSensorEventManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
// Register for events.
mSensorEventManager.registerListener(this, mSensor, SensorManager.SENSOR_DELAY_NORMAL);
// Register our receiver for the ACTION_SCREEN_OFF action. This will make our receiver
// code be called whenever the phone enters standby mode.
IntentFilter filter = new IntentFilter(Intent.ACTION_SCREEN_OFF);
registerReceiver(mReceiver, filter);
}
@Override
public void onDestroy() {
// Unregister our receiver.
unregisterReceiver(mReceiver);
// Unregister from SensorManager.
mSensorEventManager.unregisterListener(this);
}
@Override
public IBinder onBind(Intent intent) {
// We don't need a IBinder interface.
return null;
}
public void onShake() {
//Poke a user activity to cause wake?
}
public void onAccuracyChanged(Sensor sensor, int accuracy) {
//not used right now
}
//Used to decide if it is a shake
public void onSensorChanged(SensorEvent event) {
if(event.sensor.getType() != Sensor.TYPE_ACCELEROMETER) return;
Log.v("sensor","sensor change is verifying");
}
}
这个解决方法对我有用,但在我运行screebl时不起作用,这是我的很多用户真正希望与我正在开发的内容一起工作的功能。
答案 1 :(得分:1)
刚刚使用FROYO FRF50 ROM更新了我的Nexus One,过去几天一直在互联网上徘徊,如果你有一个PARTIAL_WAKELOCK,屏幕关闭的时候传感器现在可以再次运行了。我不知道这是否会成为官方版本,虽然我的理解是这是基于一些N1用户收到的官方版本。所以也许他们已经改变了主意(再次),并在手机锁定时重新启用了传感器。手指交叉。
当然,他们可以在2.2-r1再次禁用它们,谁知道......
答案 2 :(得分:1)
在HTC EVO上使用SCREEN_DIM_WAKE_LOCK
为我工作..
final PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
_WakeLock = pm.newWakeLock(PowerManager.SCREEN_DIM_WAKE_LOCK, "TAG");
_WakeLock.acquire();
答案 3 :(得分:0)
似乎这是一个2.1固件问题,我需要修复此问题,但我必须使用各种解决方法来覆盖所有版本。
我一直在整理一份工作/不工作/间歇工作的手机和固件列表,以便我们可以测试手机或固件。
http://apprssd.com/2010/09/08/android-onsensorchanged-not-working-when-screen-lock-is-on/
如果你有一部新手机,你可以更新清单,请告诉我。
答案 4 :(得分:0)
有一种替代解决方案(某种方式),您可以使用以下方式始终打开屏幕:
Settings.System.putInt(getContentResolver(), Settings.System.SCREEN_OFF_TIMEOUT,-1);
但这并不能阻止用户按下电源按钮强制关闭屏幕,然后传感器事件流停止。您还需要WRITE_SETTINGS权限才能使用。
重新注册传感器事件的上述解决方法不适用于运行Android 2.1的Droid Eris,预计不会升级。我已经使用了以下内容,虽然看起来确实有用。这种方法不是取消注册和重新注册,而是在用户将其关闭时重新打开屏幕,尽管它会变暗。在下面的代码中,handler只是一个在Activity中构造的简单Handler(即Handler handler = new Handler();)而mTurnBackOn是一个初始化为null的WakeLock。
public BroadcastReceiver mReceiver = new BroadcastReceiver() {
public void onReceive(Context context, Intent intent) {
if (Intent.ACTION_SCREEN_OFF.equals(intent.getAction())) {
// Turn the screen back on again, from the main thread
handler.post(new Runnable() {
public void run() {
if(mTurnBackOn != null)
mTurnBackOn.release();
mTurnBackOn = mPwrMgr.newWakeLock(
PowerManager.SCREEN_DIM_WAKE_LOCK |
PowerManager.ACQUIRE_CAUSES_WAKEUP,
"AccelOn");
mTurnBackOn.acquire();
}});
}
}
};