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

作者: 码农一颗颗 | 来源:发表于2017-06-25 00:43 被阅读3376次
    本篇文章是计步模块第一版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项目获取当前步数展示。


    screenshots.png
    接入方式:

    项目结构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开始计步,当天不跨天
    case_1.png
    2. 用户一直打开app计步,且跨越0点没有关闭App
    case_2.png
    3.用户打开一次App后台关闭,跨越0点且0点分隔AlertManager不能自启动(目前多数手机都是不能启动的)
    case_3.png
    4.用户打开一次app后台关闭,跨越多个0点且Alertmanager 0点分隔可以启动
    case_4.png
    5.用户开启一次app且在同一天进行重启手机(自启动不好用,很多手机不好用)
    case_5.png
    6.用户开启一次app,开关机跨0点(开机自启动不好用)
    case_6.png
    7.用户开启一次app,开关机跨0点(开机自启动可以)
    case_7.png

    缺陷

    1.方案三 跨0点打开app步数算 前一天的,如果跨越多天会导致前一天步数非常大。
    2.方案四 跨0点之前的步数会丢失(由于0点分隔AlertManager可以回调,所以可以处理0点之前的数据,以后版本在修复吧)
    3.在计步器回调中频繁调用SharePreference费电

    注意:

    1.每天早上打开App可以提高几步精度,和微信步数几乎一致。
    2.每次重启手机请打开app,会合并步数

    相关文章

      网友评论

      • o黯淡_8d2c:华为 Honor 7c 不计步,这种怎么办
      • 刘sir_418e:怎么添加点击事件 跳转 到指点的界面
        码农一颗颗:@刘sir_418e 首次安装不打开service不会启动吧
        code希必地:问个问题为什么有的手机在首次安装的时候TodayStepService就会被强杀掉
        码农一颗颗:@刘sir_418e https://www.jianshu.com/p/cfc2a200e46d
        github项目有写,只是我没更新到简书上,点击通知栏会回调个广播,你仔细看下,
        public class MyReceiver extends BaseClickBroadcast
      • 83f0c0532813:咨询,有那些机型必须有一个后台服务,才可以计步。看文档说,即使在挂起状态,也是可以计步的。想咨询一下,有那些机型是不行的,必须有一个后台服务在运行才可以计步的。
      • d01c7cc11c37:有知道如何接入微信、QQ运动步数
        码农一颗颗:@Jackhua 微信和qq没有暴露接口给客户端,但是小程序可以拿到微信的步数
      • e7e26c905d81:老哥,第4、5点现在有没研究出解决方案
      • a3f4fdbfbb5e:这个我如果用的话需要把todaystepcounterlib导入到项目里面么:scream:
        码农一颗颗:需要,你要根据你的需求修改一下,定制一下,这个只是把计步的写出来了,具体你怎么用修改一下就可以了

      本文标题:Android计步模块(类似微信运动)

      本文链接:https://www.haomeiwen.com/subject/mmgkcxtx.html