本篇文章是計(jì)步模塊第一版V1.0.0,在運(yùn)行一段時(shí)間發(fā)現(xiàn)用戶投訴很多,計(jì)步有時(shí)不準(zhǔn)確。
對(duì)計(jì)步模塊進(jìn)行優(yōu)化V2.0.0,解決大部分用戶投訴,提高計(jì)步準(zhǔn)確性,請(qǐng)查看Android計(jì)步模塊優(yōu)化(今日步數(shù)),項(xiàng)目代碼已經(jīng)更新到github,請(qǐng)點(diǎn)擊這里TodayStepCounter
最近在項(xiàng)目中研究計(jì)步模塊,每天0點(diǎn)開始記錄當(dāng)天的步數(shù),類似微信運(yùn)動(dòng)。碰到了不少坑今天有時(shí)間整理出來給大家看看。
做之前在google、baidu、github上搜了個(gè)遍沒找到好的,大多數(shù)都是需要在后臺(tái)存活,需要后臺(tái)Service。
對(duì)于現(xiàn)在的各大手機(jī)廠商為了提高電池的續(xù)航里程(省電),基本上AlertManager、android.intent.action.BOOT_COMPLETED、后臺(tái)Service都是被干掉的。
后臺(tái)保活策略Service,基本上沒什么用,被手機(jī)系統(tǒng)干掉只是時(shí)間問題,所以我認(rèn)為最好也不要去做,就算后臺(tái)存活了,用戶看到這個(gè)app非常費(fèi)電也會(huì)被刪除的。
項(xiàng)目地址:https://github.com/jiahongfei/TodayStepCounter
第二篇地址:http://www.itdecent.cn/p/c4bb1c45aeec
目前android計(jì)步有兩種方式
系統(tǒng)計(jì)步芯片
在Android4.4版本之后,部分機(jī)型實(shí)現(xiàn)了Sensor.TYPE_STEP_COUNTER傳感器,用于紀(jì)錄用戶行走的步數(shù)。從手機(jī)開機(jī)開始紀(jì)錄,手機(jī)關(guān)機(jī)時(shí)重置為0。
這個(gè)記步芯片是系統(tǒng)級(jí)別的,相對(duì)之前老版本的傳感器記步,性能有一些優(yōu)化:
不會(huì)因?yàn)锳pp單獨(dú)用了記步的功能而額外耗電
系統(tǒng)芯片記步是持續(xù)的,能夠優(yōu)化部分機(jī)型后臺(tái)不記步的問題。
加速度傳感器計(jì)算方式
加速度傳感器非常耗電,導(dǎo)致App的耗電量很高,影響用戶體驗(yàn)。
需要后臺(tái)實(shí)時(shí)運(yùn)行才能實(shí)現(xiàn)記步的功能,如果App進(jìn)程被系統(tǒng)或者安全軟件殺死,導(dǎo)致記步功能沒辦法使用
項(xiàng)目地址:https://github.com/jiahongfei/TodayStepCounter
根據(jù)以上兩種方式實(shí)現(xiàn)計(jì)步,手機(jī)提供計(jì)步傳感器就使用Sensor.TYPE_STEP_COUNTER方式(app后臺(tái)關(guān)閉也可以計(jì)步,但是部分手機(jī)也不支持,請(qǐng)看第二篇),如果不提供就使用Sensor.TYPE_ACCELEROMETER方式(app需要保持后臺(tái)運(yùn)行)。
項(xiàng)目結(jié)構(gòu):
計(jì)步Service使用單獨(dú)進(jìn)程,所以使用到進(jìn)程間通信aidl,todaystepcounterlib為庫文件用于在單獨(dú)進(jìn)程中實(shí)現(xiàn)計(jì)步算法,app依賴todaystepcounterlib項(xiàng)目獲取當(dāng)前步數(shù)展示。

接入方式:
項(xiàng)目結(jié)構(gòu)app中時(shí)如何使用計(jì)步模塊的看如下代碼
public class MainActivity extends AppCompatActivity {
private static String TAG = "MainActivity";
private static final int REFRESH_STEP_WHAT = 0;
//循環(huán)取當(dāng)前時(shí)刻的步數(shù)中間的間隔時(shí)間
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 + "步");
}
}
計(jì)步策略:
1.如果使用加速度傳感器計(jì)步必須要app在后臺(tái)存活才可以計(jì)步?!?br> 2.重頭戲是使用計(jì)步傳感器實(shí)現(xiàn)計(jì)步,app在后臺(tái)關(guān)閉也可以計(jì)步。
如下是采用Sensor.TYPE_STEP_COUNTER傳感器實(shí)現(xiàn)計(jì)步策略:
1. 用戶新安裝app,從用戶第一次打開App開始計(jì)步,當(dāng)天不跨天

2. 用戶一直打開app計(jì)步,且跨越0點(diǎn)沒有關(guān)閉App

3.用戶打開一次App后臺(tái)關(guān)閉,跨越0點(diǎn)且0點(diǎn)分隔AlertManager不能自啟動(dòng)(目前多數(shù)手機(jī)都是不能啟動(dòng)的)

4.用戶打開一次app后臺(tái)關(guān)閉,跨越多個(gè)0點(diǎn)且Alertmanager 0點(diǎn)分隔可以啟動(dòng)

5.用戶開啟一次app且在同一天進(jìn)行重啟手機(jī)(自啟動(dòng)不好用,很多手機(jī)不好用)

6.用戶開啟一次app,開關(guān)機(jī)跨0點(diǎn)(開機(jī)自啟動(dòng)不好用)

7.用戶開啟一次app,開關(guān)機(jī)跨0點(diǎn)(開機(jī)自啟動(dòng)可以)

缺陷
1.方案三 跨0點(diǎn)打開app步數(shù)算 前一天的,如果跨越多天會(huì)導(dǎo)致前一天步數(shù)非常大。
2.方案四 跨0點(diǎn)之前的步數(shù)會(huì)丟失(由于0點(diǎn)分隔AlertManager可以回調(diào),所以可以處理0點(diǎn)之前的數(shù)據(jù),以后版本在修復(fù)吧)
3.在計(jì)步器回調(diào)中頻繁調(diào)用SharePreference費(fèi)電
注意:
1.每天早上打開App可以提高幾步精度,和微信步數(shù)幾乎一致。
2.每次重啟手機(jī)請(qǐng)打開app,會(huì)合并步數(shù)