记一次Android性能优化的问题 - TextView的append方法

  最近楼主在做毕设,其中有一个评论回复的功能。在做的过程中,发现了一个问题,就是TextView在加载表情的过程非常慢。如图:


demo.gif

  作为优(闲)秀(的)码(蛋)农(疼)的我们,肯定不允许这种事情存在。所以,让我们来看看到底哪里导致了这么明显的卡顿。
  本文参考文章:

  1. 让你的EditText删除表情比微信更高效--记一次android性能分析优化实战

1. 定位问题

  由于每个人的代码不一样,所以造成卡顿的原因也是不尽相同的。本文只是记录自己在性能优化方面的一些心得和经验,只做参考。
  楼主可以从自己的代码中很明显的看出来是TextViewappend方法造成的卡顿的,楼主造成卡顿的代码如下:

public void appendContent(String content) {
        if (StringUtil.isEmpty(content)) {
            return;
        }
        final List<String> contentList = ExpressionUtil.contentToStringList(content);
        Observable.create(emitter -> {
            for (String string : contentList) {
                if (emitter.isDisposed()) {
                    emitter.onComplete();
                    return;
                }
                if (string.startsWith("#")) {
                    final String fileName = string.substring(1, string.length() - 1);
                    // 解析表情
                    Drawable drawable = ExpressionUtil.generateDrawable(getContext(), fileName);
                    if (drawable != null) {
                        // ExpressionUtil的generateImageSpannableString方法的目的是将Drawable转换成为相应的
                        // SpannableString
                        emitter.onNext(ExpressionUtil.generateImageSpannableString(drawable, fileName));
                    } else {
                        emitter.onError(new NullPointerException());
                    }
                } else {
                    emitter.onNext(string);
                }
            }
        })
                .compose(RxSchedulers.newThreadToMain())
                .subscribe(new Observer<Object>() {
                    @Override
                    public void onSubscribe(Disposable d) {
                        mDisposable = d;
                    }

                    @Override
                    public void onNext(Object o) {
                        if (o instanceof SpannableString) {
                            append((SpannableString) o);
                        } else {
                            append(o.toString());
                        }
                    }

                    @Override
                    public void onError(Throwable e) {

                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }

  如上的代码,我们很明显的可以看出来具体作用,就是在子线程里面解析表情,转化成为SpannableString,然后在主线程中通过append方法让TextView来显示表情。从这里,我们可以发现两个问题:

  1. 如果表情过多,Observable会导致背压问题。这个问题本文忽略,只是提出来。
  2. 每解析完成一个表情,就会通知主线程来渲染UI。如果表情过多,频繁的调用append方法自然而然造成卡顿。

  这里,我们不禁有一个疑问,为什么频繁调用append方法容易到底卡顿呢?我们从Android Profiler工具中得出答案,下面是我的代码抓取下来的调用栈:


  我们发现,TextViewappend方法几乎占据了CPU一半的使用时长,这太特么可怕了。
  append方法耗时的具体原因是因为每一次调用append方法都会进行layout,由于我们的表情过多,所以频繁的进行layout,自然而然会导致卡顿。关于更多详细的细节,大家可以从让你的EditText删除表情比微信更高效--记一次android性能分析优化实战这篇文章中找到答案。

2. 解决问题

  既然知道了问题的原因,解决起来自然就比较方便。频繁的调用append会导致卡顿,那我们不频繁的调用append方法就不会导致卡顿了。所以,从楼主自身的需求来看,当一个表情解析完成之后不要去通知主线程渲染,而是保存在一个SpannableStringBuilder对象,等待所有的表情解析完成之后才通知主线程调用append方法来渲染UI(本文是通过setText方法来实现):

    public void appendContent(String content) {
        if (StringUtil.isEmpty(content)) {
            return;
        }
        final List<String> contentList = ExpressionUtil.contentToStringList(content);
        Observable.create((ObservableOnSubscribe<SpannableStringBuilder>) emitter -> {
            // 初始化为TextView本来的内容
            final SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder(StringUtil.emptyIfNull(getText()));
            for (String string : contentList) {
                if (emitter.isDisposed()) {
                    emitter.onComplete();
                    return;
                }
                if (string.startsWith("#")) {
                    final String fileName = string.substring(1, string.length() - 1);
                    Drawable drawable = ExpressionUtil.generateDrawable(getContext(), fileName);
                    if (drawable != null) {
                        spannableStringBuilder.append(ExpressionUtil.generateImageSpannableString(drawable, fileName));
                    }
                } else {
                    spannableStringBuilder.append(string);
                }
            }
            // 等所有表情解析完成之后才通知主线程渲染UI
            emitter.onNext(spannableStringBuilder);
        })
                .compose(RxSchedulers.newThreadToMain())
                .subscribe(new Observer<SpannableStringBuilder>() {
                    @Override
                    public void onSubscribe(Disposable d) {
                        mDisposable = d;
                    }

                    @Override
                    public void onNext(SpannableStringBuilder spannableStringBuilder) {
                        // 调用setText方法
                        setText(spannableStringBuilder);
                    }

                    @Override
                    public void onError(Throwable e) {

                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }

  通过如上的修改,我们就会发现TextView在加载表情变得无比流畅。

3. 总结

  这是楼主第一次尝试着解决性能相关的问题,总而言之,还是收获了不少的经验。

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

推荐阅读更多精彩内容