Android 启动优化中的异步优化

一、启动优化小技巧-Theme 切换

Theme 切换启动速度没变,是让用户感觉上变快

下面是使用方式
1、创建 launcher.xml

<?xml version="1.0" encoding="utf-8"?>
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
    <!-- The background color, preferably the same as your normal theme -->
    <item android:drawable="@android:color/white"/>
    <!-- Your product logo - 144dp color version of your app icon -->
    <item>
        <bitmap android:src="@mipmap/splash"
                android:gravity="fill"/>
    </item>
</layer-list>

2、style 文件使用

<!-- Base application theme. -->
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
   <!-- Customize your theme here. -->
   <item name="colorPrimary">@color/colorPrimary</item>
   <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
   <item name="colorAccent">@color/colorAccent</item>
 </style>

<style name="Theme.Splash" parent="Theme.AppCompat.Light.NoActionBar">
   <item name="android:windowBackground">@drawable/launcher</item>
   <item name="android:windowNoTitle">true</item>
</style>

3、在 manifest 文件中使用

<activity android:name=".MainActivity"
              android:theme="@style/Theme.Splash">
    <intent-filter>
        <action android:name="android.intent.action.MAIN"/>
        <action android:name="android.intent.action.VIEW"/>

        <category android:name="android.intent.category.LAUNCHER"/>
    </intent-filter>
</activity>

4、在 Activity 中使用

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        //super.onCreate()之前切换回来
        setTheme(R.style.AppTheme)
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
    }
}

二、异步优化

核心思想:子线程分担主线程任务,并行减少执行时间

下面以 Application 为例,在 Application 中依赖了很多第三方库,下面是没有异步优化时执行的时间,如下代码所示:

class PerformanceApp : Application() {

    override fun onCreate() {
        super.onCreate()
        sApplication = this
        LaunchTimer.startRecord()
      
        initBugly()
        initUmeng()
        initFresco()
        initMap()

        LaunchTimer.endRecord("PerformApp")
    }
}

执行结果如下图所示:耗时 1370 毫秒


下面是用线程池 异步执行三方库

class PerformanceApp : Application() {

//参考 AsyncTask 源码中线程池设置的核心数,根据 CPU 的核心数确定线程池的数量
    private val CPU_COUNT = Runtime.getRuntime().availableProcessors()
    private val CORE_POOL_SIZE = Math.max(2, Math.min(CPU_COUNT - 1, 4))

    override fun onCreate() {
        super.onCreate()
        sApplication = this
        LaunchTimer.startRecord()

        val service = Executors.newFixedThreadPool(CORE_POOL_SIZE)
        service.submit {
            initBugly()
        }
        service.submit {
            initUmeng()
        }
        service.submit {
            initFresco()
        }
        service.submit {
            initMap()
        }

        LaunchTimer.endRecord("PerformApp")
    }
}

上面代码没有放在一个 service.submit { } 方法中,原因是为了减少资源浪费。理论上可以都放在一块,但是效果不够好,根据 CPU 的核心数确定线程池的数量,如果创建了 3 个,而只用了一个,这是一种浪费,所以对每个方法都 service.submit { } 一下。

执行结果如下图所示:耗时 12 毫秒


通过上面代码分析可知,异步执行效果明显。

异步优化注意:

  • 有些代码不符合异步要求(如:Handler());
  • 区分 CPU 密集型和 IO 密集型
  • 有些代码需要在某阶段完成
    如:initMap() 必须要在Application 的 onCreate() 结束掉就执行完成。
    解决方案:创建 CountDownLatch 类,代码示例如下:

class PerformanceApp : Application() {

    //参考 AsyncTask 源码中线程池设置的核心数,根据 CPU 的核心数确定线程池的数量
    private val CPU_COUNT = Runtime.getRuntime().availableProcessors()
    private val CORE_POOL_SIZE = Math.max(2, Math.min(CPU_COUNT - 1, 4))

    private val mCountDownLatch = CountDownLatch(1)

   
    override fun onCreate() {
        super.onCreate()
        sApplication = this
        LaunchTimer.startRecord()
        val service = Executors.newFixedThreadPool(CORE_POOL_SIZE)
        service.submit {
            initBugly()
        }
        service.submit {
            initUmeng()
        }
        service.submit {
            initFresco()
        }
        service.submit {
            //高德地图
            initMap()
            mCountDownLatch.countDown()
        }
        //CountDownLatch不被满足的话会一直等待
        mCountDownLatch.await()
        LaunchTimer.endRecord("PerformApp")
    }
}

缺点:

  1. 代码不够优雅
  2. 场景不好处理(依赖关系)
  3. 特定时间段内结束某个任务不好处理
  4. 维护成本高

CountDownLatch 参考
java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例
CountDownLatch详解

三、异步优化启动器

核心思想:充分利用CPU多核,自动梳理任务顺序
启动器流程

  • 代码 Task 化,启动逻辑抽象为 Task
  • 根据所有任务依赖关系排序生成一个有向无环图
  • 多线程按照排序后的优先级依次执行

headTask:所有Task执行之前所做的事情
tailTask:所有任务结束之后
idleTask:程序空闲时执行

使用示例:

class PerformanceApp : Application() {

      override fun onCreate() {
        super.onCreate()
        sApplication = this
        LaunchTimer.startRecord()
        TaskDispatcher.init(this)
        val dispatcher = TaskDispatcher.createInstance()
        dispatcher.addTask(InitAMapTask())
            .addTask(InitStethoTask())
            .addTask(InitWeexTask())
            .addTask(InitBuglyTask())
            .addTask(InitFrescoTask())
            .addTask(InitJPushTask())
            .addTask(InitUmengTask())
            .addTask(GetDeviceIdTask())
            .start()
        //启动器中配置需要等待的函数没有完成是都会等待
        dispatcher.await()
        LaunchTimer.endRecord("task")
   }
}

四、延迟初始化方案

1、常规方案
  • new Handler().postDelayed
  • Feed 展示后调用

示例展示:
在 NewsAdapter 中设置数据展示后的监听器

//NewsAdapter 中
public void setOnFeedShowCallBack(OnFeedShowCallBack callBack) {
        this.mCallBack = callBack;
}
@Override
    public void onBindViewHolder(@NonNull final ViewHolder holder, int position) {
        //onBindViewHolder 回调多次,而只统计一次,加个变量标识
        if (position == 0 && !mHasRecorded) {
            mHasRecorded = true;
            holder.layout.getViewTreeObserver()
                    .addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
                        @Override
                        public boolean onPreDraw() {
                            holder.layout.getViewTreeObserver().removeOnPreDrawListener(this);
                            LaunchTimer.Companion.endRecord("FeedShow");
                            if (mCallBack != null) {
                                mCallBack.onFeedShow();
                            }
                            return true;
                        }
                    });
        }
}

//MainActivity 中
override fun onFeedShow() {
    DispatchRunnable(DelayInitTaskA()).run()
    DispatchRunnable(DelayInitTaskB()).run()
}

问题:

  • 导致 Feed 卡顿:我们 NewsAdapter 中的回调是在主线程中执行,那么 MainActivity 中的 onFeedShow() 异步执行可能会延迟,造成卡顿
  • 时间不便控制
2、优化方案

核心思想:对延迟任务进行分批初始化

  • 利用 IdleHandler 特性,空闲执行

示例展示:
首先创建 DelayInitDispatcher 类

public class DelayInitDispatcher {

    private Queue<Task> mDelayTasks = new LinkedList<>();
    /**
     * IdleHandler:在系统空闲后执行
     */
    private MessageQueue.IdleHandler mIdleHandler = new MessageQueue.IdleHandler() {
        //系统空闲时回调
        @Override
        public boolean queueIdle() {
            if (mDelayTasks.size() > 0) {
                Task task = mDelayTasks.poll();
                new DispatchRunnable(task).run();
            }
            return !mDelayTasks.isEmpty();
        }
    };

    public DelayInitDispatcher addTask(Task task) {
        mDelayTasks.add(task);
        return this;
    }

    public void start() {
        Looper.myQueue().addIdleHandler(mIdleHandler);
    }

}

创建 DispatchRunnable 类

/**
 * 任务真正执行的地方
 */

public class DispatchRunnable implements Runnable {
    private Task mTask;
    private TaskDispatcher mTaskDispatcher;

    public DispatchRunnable(Task task) {
        this.mTask = task;
    }
    public DispatchRunnable(Task task,TaskDispatcher dispatcher) {
        this.mTask = task;
        this.mTaskDispatcher = dispatcher;
    }

    @Override
    public void run() {
        TraceCompat.beginSection(mTask.getClass().getSimpleName());
        DispatcherLog.i(mTask.getClass().getSimpleName()
                + " begin run" + "  Situation  " + TaskStat.getCurrentSituation());

        Process.setThreadPriority(mTask.priority());

        long startTime = System.currentTimeMillis();

        mTask.setWaiting(true);
        mTask.waitToSatisfy();

        long waitTime = System.currentTimeMillis() - startTime;
        startTime = System.currentTimeMillis();

        // 执行Task
        mTask.setRunning(true);
        mTask.run();

        // 执行Task的尾部任务
        Runnable tailRunnable = mTask.getTailRunnable();
        if (tailRunnable != null) {
            tailRunnable.run();
        }

        if (!mTask.needCall() || !mTask.runOnMainThread()) {
            printTaskLog(startTime, waitTime);

            TaskStat.markTaskDone();
            mTask.setFinished(true);
            if(mTaskDispatcher != null){
                mTaskDispatcher.satisfyChildren(mTask);
                mTaskDispatcher.markTaskDone(mTask);
            }
            DispatcherLog.i(mTask.getClass().getSimpleName() + " finish");
        }
        TraceCompat.endSection();
    }

    /**
     * 打印出来Task执行的日志
     *
     * @param startTime
     * @param waitTime
     */
    private void printTaskLog(long startTime, long waitTime) {
        long runTime = System.currentTimeMillis() - startTime;
        if (DispatcherLog.isDebug()) {
            DispatcherLog.i(mTask.getClass().getSimpleName() + "  wait " + waitTime + "    run "
                    + runTime + "   isMain " + (Looper.getMainLooper() == Looper.myLooper())
                    + "  needWait " + (mTask.needWait() || (Looper.getMainLooper() == Looper.myLooper()))
                    + "  ThreadId " + Thread.currentThread().getId()
                    + "  ThreadName " + Thread.currentThread().getName()
                    + "  Situation  " + TaskStat.getCurrentSituation()
            );
        }
    }

}

在 MainActivity 中的使用

 override fun onFeedShow() {
        val delayInitDispatcher = DelayInitDispatcher()
        delayInitDispatcher.addTask(DelayInitTaskA())
            .addTask(DelayInitTaskB())
            .start()
 }

优点:

  • 执行时机明确
  • 缓解 Feed 卡顿
    每次只执行一个 Task,而且是在系统空闲时执行。

实例 demo 如:PerformanceDemo

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

推荐阅读更多精彩内容