Android TextView使用及性能优化

Android TextView使用及性能优化

TextView 是Android中最常用的控件,在这里记录下TextView 的用法;

字体

系统自带字体

在Android中可以使用系统自带的4种字体:

  1. normal 普通字体,系统默认使用的字体
  2. sans 非衬线字体
  3. serif 衬线字体
  4. monospace 等宽字体

在XML中使用android:typeface="normal"进行设置

自定义字体

将字体文件放到main/assets/fonts目录下,使用Asset读取字体后进行设置

AssetManager assetManager = mContext.getAssets();
Typeface typeface = Typeface.createFromAsset(assetManager, "fonts/SourceCodePro-Bold.otf");
textView.setTypeface(typeface);

Drawable

使用android:drawableLeft="@mipmap/ic_launcher"可以设置一张图片显示在文字的上下左右,减少布局层级

Span

使用Span能够在一段TextView中设置不同颜色的字体,链接,图片等内容

ClickableSpan

使用ClickableSpan 能够设置一段文字的点击事件

创建自己的MyClickableSpan:

public class MyClickableSpan extends ClickableSpan {
    @Override
    public void onClick(View view) {
        //点击的回调
    }

    @Override
    public void updateDrawState(TextPaint ds) {
        super.updateDrawState(ds);
        //可以在这里通过ds来设置可点击文字的样式
        ds.setColor(Color.RED);
        ds.setUnderlineText(false);
    }
}

之后使用SpannableStringBuilder来创建字符串,并使用setSpan来为字符串的一部分设置Span对象

SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder("This Is Test Text, Click This");
spannableStringBuilder.setSpan(
    new MyClickableSpan(),      //span对象
    0, 5,                       //开始和结束字符index
    Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);//标识
mTextView.setMovementMethod(LinkMovementMethod.getInstance());  //不设置无法响应点击事件
mTextView.setText(spannableStringBuilder);

其中setSpan()方法的最后一个参数标识有以下常量,这些常量标识着在对SpannableStringBuilder进行insert时添加的字符适用的规则:

Spanned.SPAN_EXCLUSIVE_EXCLUSIVE

Spanned.SPAN_EXCLUSIVE_INCLUSIVE

Spanned.SPAN_INCLUSIVE_EXCLUSIVE

Spanned.SPAN_INCLUSIVE_INCLUSIVE

前一个EXCLUSIVE/INCLUSIVE标识着在设置了Span的一段字符之前(紧挨着)插入字符时,被不被包含到Span范围中,EXCLUSIVE表示包含,INCLUSIVE表示不包含;

第二个EXCLUSIVE/INCLUSIVE同理表示插入这段字符之后的效果;

ImageSpan

ImageSpan用于在TextView中插入图片,可以用来实现图文混排

使用方法:

SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder("This Is Test Text, Click This");
//创建Span对象
ImageSpan imageSpan = new ImageSpan(this, BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher));
//设置ImageSpan,在第3和第5个字符中间的字符将会被替换成图片
spannableStringBuilder.setSpan(imageSpan, 3, 5, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
mTextView.setText(spannableStringBuilder);

这样实现的效果是文字与图片底部进行对齐,如果需要图片中线与文字中线对其,需要自己重写ImageSpan

TextView性能优化

介绍:

​ Android 中的TextView中存在着很多EditText中的特性,在setText()方法中会涉及到很多Span相关的操作,比如设置TextWatcher,重新构造Spannable等操作,在我们仅仅显示静态文本的时候这些操作都是没有必要的(通过使用普通的TextView进行Debug来验证普通的TextView的确是Span的);

​ 在大量显示静态文本的时候就可以通过StaticLayout来计算出TextView的布局信息,这项工作可以放到非UI线程来进行,能够减少在setText()的时候UI线程的耗时,达到优化TextView性能的目的;

​ StaticLayout是TextView中用于显示多行静态文本的Layout,也是能够支持SpannableString的,只是不能在Span变化之后重新Layout,所以在大部分场景下已经适用;

使用方法:

  1. 自定义View
public class StaticLayoutView extends View {
    
    private Layout layout = null;
    
    private int width;
    private int height;
    
    public void setLayout(Layout layout) {
        this.layout = layout;
        if (this.layout.getWidth() != width || this.layout.getHeight() != height) {
            width = this.layout.getWidth();
            height = this.layout.getHeight();
            requestLayout();
        }
    } 

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        
        canvas.save();
        if (layout != null) {
            //直接使用layout来绘制文本
            layout.draw(canvas, null, null, 0);
        }
        canvas.restore();
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        super.onLayout(changed, left, top, right, bottom);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        if (layout != null) {
            setMeasuredDimension(layout.getWidth(), layout.getHeight());
        } else {
            super.onMeasure(widthMeasureSpec, heightMeasureSpec);   
        }
        
    }
}

通过这个自定义的View来显示Text,在onDraw()的时候直接使用layout来进行绘制,而设置需要显示的文本则直接使用setLayout()来实现

  1. 创建StaticLayout

            StaticLayout layout = new StaticLayout(
                    text,            //文本
                    textPaint,       //TextPaint对象
                    layoutWidth, //layout宽度,多行的情况下取父容器宽度减去padding即可
                    Layout.Alignment.ALIGN_NORMAL, //对其方式
                    1.0f, //间隔倍数,表示1.0倍字体大小
                    0f,   //增加的间隔
                    true
            );
    
  2. 在Android 4.0之后增加了TextLayoutCache对文本的测量和渲染结果进行缓存,于是可以在创建了StaticLayout对象之后就调用其draw()方法来给缓存增加本次的测量结果

            Canvas canvas = new Canvas();
            staticLayout.draw(canvas);
    

    这些操作都应该放到非UI线程来完成,比如在文本下载完成之后就生成对应的StaticLayout,之后在TextView要显示的时候setLayout()即可

性能测试

使用下面给出的参考链接中的测试Demo在 ZTE A2017 Android7.1.1 高通820设备上,普通TextView在ListView中连续滚动的帧数是55帧,使用StaticLayout的结果为60帧

可以作为在APP使用CPU资源较多的情况下的优化手段

参考链接:TextView预渲染研究

Android P中的TextView新特性

在Android中,TextView的测量消耗了大量的时间,Android P中提供了PrecomputedText能够将测量这个过程放到后台来执行,减轻对于UI线程的卡顿;

使用方法:

非Android P时,使用AppCompatTextView控件,使用setTextFeature()方法来将文本的measure过程放到其他线程来执行,而不是直接将text应用于TextView;

在调用了这个方法之后如果对TextView进行边距,文字大小等的设置都将会报错;

        textView.setTextFuture(
                PrecomputedTextCompat.getTextFuture(
                    text,  //文本
                    textView.getTextMetricsParamsCompat(), //PrecomputedTextCompat.Params
                    executor) //线程池,
        );

Prefetch Text Layout in RecyclerView

PrecomputedTextCompat

性能测试

在ListView中仅替换设置Text的方法时未测试出性能与普通方法有什么优势,猜测是ListView没有在getView和显示之间预留时间,

测试项目地址:

https://github.com/GavynZhang/PrecomuptedTextViewTest

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

推荐阅读更多精彩内容