分享

Android 计步模块(类似微信运动)

 codingSmart 2021-10-22

前言

目前 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,会合并步数

与之相关

Android 选择图片上传功能【支持多选拍照预览等】

Android 模仿微信选择图片

关键词:code小生

    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多