最近在项目中研究计步模块,每天0点开始记录当天的步数,类似微信运动。碰到了不少坑今天有时间整理出来给大家看看。
做之前在google、baidu、github上搜了个遍没找到好的,大多数都是需要在后台存活,需要后台Service。

对于现在的各大手机厂商为了提高电池的续航里程(省电),基本上AlertManager、

android.intent.action.BOOT_COMPLETED、后台Service都是被干掉的。

后台保活策略Service,基本上没什么用,被手机系统干掉只是时间问题,所以我认为最好也不要去做,就算后台存活了,用户看到这个app非常费电也会被删除的。

项目地址:https://github.com/jiahongfei/TodayStepCounter

目前android计步有两种方式

系统计步芯片

在Android4.4版本之后,部分机型实现了Sensor.TYPE_STEP_COUNTER传感器,用于纪录用户行走的步数。从手机开机开始纪录,手机关机时重置为0。

这个记步芯片是系统级别的,相对之前老版本的传感器记步,性能有一些优化:

不会因为App单独用了记步的功能而额外耗电

系统芯片记步是持续的,能够优化部分机型后台不记步的问题。

加速度传感器计算方式

加速度传感器非常耗电,导致App的耗电量很高,影响用户体验。
需要后台实时运行才能实现记步的功能,如果App进程被系统或者安全软件杀死,导致记步功能没办法使用

项目地址:https://github.com/jiahongfei/TodayStepCounter

根据以上两种方式实现计步,手机提供计步传感器就使用Sensor.TYPE_STEP_COUNTER方式(app后台关闭也可以计步),如果不提供就使用SensorManager.SENSOR_DELAY_UI方式(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,会合并步数

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

02-02 12:21