App启动优化,一种较优秀的方案


项目重构App的启动,之前的处理是使用SplashActivity作为过渡页,然后将Application中的第三方SDK放到IntentService中加载,启动过程中的一些耗时操作也都去掉了,奈何MainActivity中的业务逻辑还是太过复杂,启动仍然比较漫长。研究了一下淘宝项目的启动处理,总结出一种非常优化的App启动方案。

传统的启动处理

市面上大部分的App启动处理如出一辙,基本都是先进入SplashActivity,然后Handler延时后跳转到MainActivity。这样做的好处是SplashActivity作为过渡页更加轻量,启动到显示界面的时间更短,给用户较好的体验,而且SplashActivity中也可以提前做一些数据处理,减轻MainActivity的压力。但是加入SplashActivity过渡页也有一些缺点,比如要多绘制一个页面,要使用Intent传递数据,不能减轻MainActivity页面的绘制压力和一些数据加载。


传统的启动处理

除此之外,Application的处理也是一个很重要的点,因为App启动是先走Application的onCreate方法的,如果onCreate中加载SDK的时间过于漫长,就会导致App启动后显示一段时间白屏页才会进入SplashActivity,当然也可以使用windowBackGround属性避免白屏,但是时间长了总是会影响用户体验的。所以Application中一般都是采用IntentService来加载SDK的。

为了解决App启动慢和SplashActivity过渡启动的缺点,个人去研究了一下淘宝的启动,发现第一次启动耗时大概3秒,然后退出去,再进入时就不会显示Splash,直接进入主页读取缓存显示页面。我猜测淘宝的启动肯定是根据应用的进程是否存活,如果存活就直接进入MainActivity,否则显示Splash,而且淘宝的Splash加载很快,MainActivity的加载也很快,所以淘宝应该是采用了Splash绑定在Mainactivity上的方式,即SplashFragment+StubView的方式启动App。

优化后的Splash

传统的App启动方式不足以满足现在用户的体验,用户需要的是加载更快,性能更优的体验,所以这里采用SplashFragment+StubView的方式去实现App的快速启动。


优化后的Splash

跟传统启动方式最大的不同是,优化后的启动会直接进入MainActivity,然后在MainActivity控制Splash的显示隐藏,这样在填充Splash的时候还以为直接加载MainActivity的数据,当窗体填充完毕之后,就可以绘制MainActivity的布局,然后移除Splash,展示页面,这种方式很好的体现了异步的操作,使得填充和加载分离,启动的速度更快。

既然知道了这种实现方式,那具体代码是如何编写的呢,接下来看一下如何实现。

1. 填充SplashFragment

在MainActivity的onCreate方法中,首先,需要做的就是填充Splash并显示,显示的内容就可以在SplashFragment中编辑。

    log.d(TAG, "==========填充SplashFragment==========");
    // 1.初始化SplashFragment,填充Splash
    final SplashFragment splashFragment = new SplashFragment();
    getFragmentManager().beginTransaction().replace(R.id.container,splashFragment).commitAllowingStateLoss();

2. 窗体部署完毕,填充主页布局

当整个页面最底层DecorView填充完毕之后,就可以开始去填充MainActivity的布局了,并且使用Handler发送延时消息,2秒后移除Splash。当然移除Splash也可以直接在数据加载完毕显示页面之前移除掉。

    getWindow().getDecorView().post(new Runnable() {
        @Override
        public void run() {
            // 填充布局
            viewStub.inflate();
            // 初始化布局
            initView();
            // 2秒后移除Splash
            mHandler.postDelayed(new DelayRunnable(MainActivity.this, splashFragment), 2000);
        }
    });

3. 加载MainActivity数据

其实加载数据会在第二步之前执行,因为第二步是异步操作,并且加载数据是耗时操作,所以需要更早执行,不要放在第二步里去执行,这里在测试的时候有可能数据先加载完毕,布局才绘制完,调用控件就会有空指针问题,但是实际开发中肯定不会出现这种情况的,MainActivity中的数据加载肯定很耗时。

Application的优化

Application是应用的核心枢纽,应用启动时会先走Application的onCreate方法去初始化一些全局数据,比如SDK等等。但是当全局数据很多时,App的启动就会变得缓慢,因为初始化是同步加载的,所以要想加快启动,就需要将延时操作放到后台去处理。

Android里有一个Service,正好可以做延时处理的操作,这个Service就是IntentService。关于IntentService,不了解的看这里。
IntentService 示例与详解
IntentService文档

接下来,看一下如何实现。

  • 自定义InitializeService继承IntentService,实现构造和方法onHandleIntent

  • 在Application的onCreate方法中启动初始化服务

     // 启动服务去做耗时操作
     InitializeService.start(this);

     public static void start(Context context) {
           Intent intent = new Intent(context, InitializeService.class);
            intent.setAction(ACTION_APP_LAUNCHER);
            context.startService(intent);
     }
  • 在onHandleIntent方法中判断Intent,然后去初始化SDK
     @Override
      protected void onHandleIntent(@Nullable Intent intent) {
          if (intent != null) {
                final String action = intent.getAction();
                if (ACTION_APP_LAUNCHER.equals(action)) {
                    performInit();
                }
          }
      }
    
    
        /**
         * 启动初始化耗时操作
         */
        private void performInit() {
            // 模拟延时加载
            SystemClock.sleep(2000);
            Log.d(TAG, "==========初始化第三方SDK结束==========");
         }

注意:

  • 在清单文件中配置Application和Service
  • 并不是所有的SDK都能放到Service中去初始化的,在MainActivity中用到的还是需要提前初始化

Demo地址


欢迎大家访问我的简书博客GitHub

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,099评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,828评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,540评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,848评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,971评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,132评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,193评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,934评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,376评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,687评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,846评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,537评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,175评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,887评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,134评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,674评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,741评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,892评论 25 707
  • 转自 1. 什么是Activity? 四大组件之一,一般的,一个用户交互界面对应一个activity setCon...
    joe1632阅读 1,392评论 0 7
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,642评论 18 139
  • 谷歌官方文档:https://developer.android.com/topic/performance/la...
    路Promenade阅读 2,367评论 18 15
  • 行走在无边无际的原野 轻风佛面 金色麦浪摇曳着它动人的身躯 小溪潺潺 鱼儿轻摇着它柔软的腰肢 歌声阵阵 老农哼唱着...
    月夜醉听风阅读 255评论 1 2