Android冷启动实现APP秒开

作者: 李晨玮 | 来源:发表于2016-08-28 16:47 被阅读13635次

    在阅读这篇文章之前,首先需要理解几个东西:
    1、什么是Android的冷启动时间?
    冷启动时间是指用户从手机桌面点击APP的那一刻起到启动页面的Activity调用onCreate()方法之间的这个时间段。
    2、在冷启动的时间段内发生了什么?
    首先我们要知道当打开一个Activity的时候发生了什么,在一个Activity打开时,如果该Activity所属的Application还没有启动,那么系统会为这个Activity创建一个进程(每创建一个进程都会调用一次Application,所以Application的onCreate()方法可能会被调用多次),在进程的创建和初始化中,势必会消耗一些时间,在这个时间里,WindowManager会先加载APP里的主题样式里的窗口背景(windowBackground)作为预览元素,然后才去真正的加载布局,如果这个时间过长,而默认的背景又是黑色或者白色,这样会给用户造成一种错觉,这个APP很卡,很不流畅,自然也影响了用户体验。

    来看下效果图:

    未优化 优化方案1 优化方案2

    一、消除启动时的白屏/黑屏

    在用户点击手机桌面APP的时候,看到的黑屏或者白屏其实是界面渲染前的第一帧,如果你看懂了文章头的那2个问题,那么解决这个问题就非常轻松了,无非就是将Theme里的windowBackground设置成我们想要让用户看到的画面就可以了,这里有2种做法:

    1、将背景图设置成我们APP的Logo图,作为APP启动的引导,现在市面上大部分的APP也是这么做的。

        <style name="AppWelcome" parent="AppTheme">
            <item name="android:windowBackground">@mipmap/bg_welcome_start</item>
        </style>
    

    2、将背景颜色设置为透明色,这样当用户点击桌面APP图片的时候,并不会"立即"进入APP,而且在桌面上停留一会,其实这时候APP已经是启动的了,只是我们心机的把Theme里的windowBackground的颜色设置成透明的,强行把锅甩给了手机应用厂商(手机反应太慢了啦,哈哈),其实现在微信也是这样做的,不信你可以试试。

        <style name="Appwelcome" parent="android:Theme.Translucent.NoTitleBar.Fullscreen"/>
    

    透明化这种做法需要注意的一点,如果直接把Theme引入Activity,在运行的时候可能会出现如下异常:

    java.lang.IllegalStateException: You need to use a Theme.AppCompat theme (or descendant) with this activity.

    这个是因为使用了不兼容的Theme,例如我这里的Activity继承了AppCompatActivity,解决方案很简单:
    1、让其Activity集成Activity而不要集成兼容性的AppCompatActivity
    2、在onCreate()方法里的super.onCreate(savedInstanceState)之前设置我们原来APP的Theme

    public class MainActivity extends AppCompatActivity {
        @Override
        protected void onCreate(Bundle savedInstanceState) {
                setTheme(R.style.AppTheme);
                super.onCreate(savedInstanceState);
        }
    }
    

    上面的2种做法,我们都需要将Theme引入对应的Activity

            <activity
                android:name=".app.main.MainActivity"
                android:theme="@style/AppWelcome"
                android:screenOrientation="portrait">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
    

    二、关于启动优化

    上面的做法其实可以达到"秒开"APP的效果,不过确不是真实的速度,在Activity创建过程中其实是会经过一系列framework层的操作,在日常开发中,我们都会去重写Application类,然后在Application里进行一些初始化操作,比如存放用户标识的静态化TOKEN,第三方SDK的初始化等。
    这里给出几点建议:
    1、不要让Application参与业务的操作
    2、不要在APPlication进行耗时操作,比如有些开发者会在自己的APP里一系列文件夹或文件(比如我自己),这些I/O操作应该放到"确实该使用的时候再去创建"亦或者是数据库的一些操作。
    3、不要以静态变量的方式在Application中保存数据等。

    当然这是绝对的理想主义,把上面的"不要"2字之前添上"尽量"2字吧,毕竟在实际开发中,这样做确实会让我们方便许多。

    对了,补充一点,布局也是很重要的,尽量的去减少布局的复杂性,布局深度,因为在View绘制的过程中,测量也是很耗费性能的。

    相关文章

      网友评论

      • 深爱着伊豆的流云:建议你试试layer-list
      • lsande:冷启动,有开屏3秒的广告,如何把加载广告的时间给去掉。加一个sleep 时间么?
      • cheetah747:对于非开发人员有没有什么办法可以优化冷启动的方案??????我记得我手机之前冷启动都挺快的,现在越来越慢,刷机也没用。。。。
        李晨玮:最简单的方式提高手机配置。。现在的App越来越大,越来越占内存了
      • zzl93:windowBackground 设置的是background 会拉伸logo图片 怎么办。 (在splash中用的是image的src来加载logo图)
        zzl93:@深爱着伊豆的流云 好的我试试i
        深爱着伊豆的流云:@zzl93 建议您试试layer-list
      • 697b90ed256b:好纠结。。。。我测了一下就算是一个初始软件,在android上启动也会有那么1秒的白屏,而且我手机还是千元机,感情这个问题在于android啊。这什么概念,表示对android系统很失望。
        那些什么设置背景图片肯定不行的啊,不同的业务逻辑,有的启动页必须要接入广告等东西,而且图片还要设配设备。
      • AIllll:设置了windowBackground之后,有些界面背景就会变成设置的这个图片,怎么办,能在启动之后动态设置回去吗
        李晨玮:@AIllll 不需要的,这种带windowBackground背景图的风格,你只需要在APP启动页上单独设置就可以,比如:
        <activity
        android:name=".app.common.view.activity.CommonWelcomeActivity"
        android:screenOrientation="portrait"
        android:theme="@style/AppWelcome">
        <intent-filter>
        <action android:name="android.intent.action.MAIN"/>
        <category android:name="android.intent.category.LAUNCHER"/>
        </intent-filter>
        </activity>
      • 30ea29ca3cf1:是帅哥嘛?不是的话我不看
      • MISSGY_:我们是在splashactivity中做耗时操作,做不完首页进不去,所以网络卡会一直卡在splash页面
      • 367053bacd51:第一种方法如果是设置了intentFlag为Intent.FLAG_ACTIVITY_CLEAR_TASK | Intent.FLAG_ACTIVITY_NEW_TASK之后,每次销毁splash打开mainActivity还会显示一次windowbackground
      • 丶麦芽:请问一下为什么不要以静态变量的方式在Application中保存数据?不太理解。
        丶麦芽:@李晨玮 谢谢:+1: :+1:
        李晨玮: @丶麦芽 在实际开发中这种情况是很常见的,有些时候确实是没法避免,只能说尽量做到,这里涉及到内存占用问题,静态变量在编译运行的时候就开辟了内存空间并有引用指向它,这块内存是不会被gc回收的,会一直存在。
      • xiawe_i:小伙,不错哦,主要还是Application里初始化的东西太多,可以挪到Splash里.
      • 叶落竹影:第二种方法学习了
      • 男子汉大豆腐:挺好,比较新颖的思路。
      • yunnywu:第二种不需要那么麻烦 继承你的主题 设置窗口背景透明就行了
      • a09b39bda0fc:为何我的设置windowBackground为启动图 然后冷启动时会显示黑色的状态栏 而大神你的是直接全屏呢?我启动页设置了全屏,这与我启动页的启动图相差一个状态栏的高度
        李晨玮: @今晚会有雨_ff3d 嗯,成功了就好
        a09b39bda0fc:@李晨玮 parent="@android:style/Theme.NoTitleBar.Fullscreen" 是这样吧? 我试了下这样是成功全屏的
        李晨玮:这里需要多一个风格设置,注意全屏Fullscreen。
      • 年才下:所以说大多APP在启动页面卡很久的原因是因为Application的耗时操作写多了吗?
        李晨玮:这个是综合因素导致的,也不一定是Application里的耗时操作写多了。
      • 菜鸟Android:楼主,总结的不错!!!
      • 丰富的单纯:直接在主题中设置android:windowBackground为启动图片,这个不好适配,得做多张图片
      • 898d7d74122c:要越狱吗
        李晨玮::sweat: Android里应该叫Root,这个不需要的哈。
      • 0e320bd3c18d:楼主,我按照你的方式做了一下,通过home移除应用,点击app图标,启动还是很慢,这有办法解决吗?
        李晨玮:@624109061 你是什么开发工具呢,如果是android studio的话,2.1还是2.2版本后在debug调试下还是会有这个问题的,你正式打包的时候就不会。
      • 捡淑:66666
      • 乘风破浪的程序员:首页设置了个背景颜色为白色
      • 乘风破浪的程序员:设置背景background 后进入maniactiviy 整个页面变成了图片背景
      • 代码咖啡:思路很棒哦 学习了!
        李晨玮:@iNerdStack :smile: 加油
      • SundyXu:可以,这个很可以
      • 叶舞清风:这些学习了
      • 华清松:马克
      • 74d9e05e849b:666
        LPhoenix:我也想

      本文标题:Android冷启动实现APP秒开

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