本篇文章是计步模块第一版V1.0.0,在运行一段时间发现用户投诉很多,计步有时不准确。
对计步模块进行优化V2.0.0,解决大部分用户投诉,提高计步准确性,请查看Android计步模块优化(今日步数),项目代码已经更新到github,请点击这里TodayStepCounter
最近在项目中研究计步模块,每天0点开始记录当天的步数,类似微信运动。碰到了不少坑今天有时间整理出来给大家看看。
做之前在google、baidu、github上搜了个遍没找到好的,大多数都是需要在后台存活,需要后台Service。
对于现在的各大手机厂商为了提高电池的续航里程(省电),基本上AlertManager、android.intent.action.BOOT_COMPLETED、后台Service都是被干掉的。
后台保活策略Service,基本上没什么用,被手机系统干掉只是时间问题,所以我认为最好也不要去做,就算后台存活了,用户看到这个app非常费电也会被删除的。
项目地址:https://github.com/jiahongfei/TodayStepCounter
第二篇地址:http://www.jianshu.com/p/c4bb1c45aeec
目前android计步有两种方式
系统计步芯片
在Android4.4版本之后,部分机型实现了Sensor.TYPE_STEP_COUNTER传感器,用于纪录用户行走的步数。从手机开机开始纪录,手机关机时重置为0。
这个记步芯片是系统级别的,相对之前老版本的传感器记步,性能有一些优化:
不会因为App单独用了记步的功能而额外耗电
系统芯片记步是持续的,能够优化部分机型后台不记步的问题。
加速度传感器计算方式
加速度传感器非常耗电,导致App的耗电量很高,影响用户体验。
需要后台实时运行才能实现记步的功能,如果App进程被系统或者安全软件杀死,导致记步功能没办法使用
项目地址:https://github.com/jiahongfei/TodayStepCounter
根据以上两种方式实现计步,手机提供计步传感器就使用Sensor.TYPE_STEP_COUNTER方式(app后台关闭也可以计步,但是部分手机也不支持,请看第二篇),如果不提供就使用Sensor.TYPE_ACCELEROMETER方式(app需要保持后台运行)。
项目结构:
计步Service使用单独进程,所以使用到进程间通信aidl,todaystepcounterlib为库文件用于在单独进程中实现计步算法,app依赖todaystepcounterlib项目获取当前步数展示。
接入方式:
项目结构app中时如何使用计步模块的看如下代码
public class MainActivity extends AppCompatActivity {
private static String TAG = "MainActivity";
private static final int REFRESH_STEP_WHAT = 0;
//循环取当前时刻的步数中间的间隔时间
private long TIME_INTERVAL_REFRESH = 500;
private Handler mDelayHandler = new Handler(new TodayStepCounterCall());
private int mStepSum;
private ISportStepInterface iSportStepInterface;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent(this, VitalityStepService.class);
startService(intent);
bindService(intent, new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
iSportStepInterface = ISportStepInterface.Stub.asInterface(service);
try {
mStepSum = iSportStepInterface.getCurrTimeSportStep();
updateStepCount();
} catch (RemoteException e) {
e.printStackTrace();
}
mDelayHandler.sendEmptyMessageDelayed(REFRESH_STEP_WHAT, TIME_INTERVAL_REFRESH);
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
}, Context.BIND_AUTO_CREATE);
}
class TodayStepCounterCall implements Handler.Callback{
@Override
public boolean handleMessage(Message msg) {
switch (msg.what) {
case REFRESH_STEP_WHAT: {
if (null != iSportStepInterface) {
int step = 0;
try {
step = iSportStepInterface.getCurrTimeSportStep();
} catch (RemoteException e) {
e.printStackTrace();
}
if (mStepSum != step) {
mStepSum = step;
updateStepCount();
}
}
mDelayHandler.sendEmptyMessageDelayed(REFRESH_STEP_WHAT, TIME_INTERVAL_REFRESH);
break;
}
}
return false;
}
}
private void updateStepCount() {
Log.e(TAG,"updateStepCount : " + mStepSum);
TextView stepTextView = (TextView)findViewById(R.id.stepTextView);
stepTextView.setText(mStepSum + "步");
}
}
计步策略:
1.如果使用加速度传感器计步必须要app在后台存活才可以计步。‘
2.重头戏是使用计步传感器实现计步,app在后台关闭也可以计步。
如下是采用Sensor.TYPE_STEP_COUNTER传感器实现计步策略:
1. 用户新安装app,从用户第一次打开App开始计步,当天不跨天
2. 用户一直打开app计步,且跨越0点没有关闭App
3.用户打开一次App后台关闭,跨越0点且0点分隔AlertManager不能自启动(目前多数手机都是不能启动的)
4.用户打开一次app后台关闭,跨越多个0点且Alertmanager 0点分隔可以启动
5.用户开启一次app且在同一天进行重启手机(自启动不好用,很多手机不好用)
6.用户开启一次app,开关机跨0点(开机自启动不好用)
7.用户开启一次app,开关机跨0点(开机自启动可以)
缺陷
1.方案三 跨0点打开app步数算 前一天的,如果跨越多天会导致前一天步数非常大。
2.方案四 跨0点之前的步数会丢失(由于0点分隔AlertManager可以回调,所以可以处理0点之前的数据,以后版本在修复吧)
3.在计步器回调中频繁调用SharePreference费电
注意:
1.每天早上打开App可以提高几步精度,和微信步数几乎一致。
2.每次重启手机请打开app,会合并步数