在理解View绘制原理之前,我们先要明白View是什么。
Android中是这样理解View这个类的

* This class represents the basic building block for user interface components. A View
 * occupies a rectangular area on the screen and is responsible for drawing and
 * event handling. View is the base class for <em>widgets</em>, which are
 * used to create interactive UI components (buttons, text fields, etc.). The
 * {@link android.view.ViewGroup} subclass is the base class for <em>layouts</em>, which
 * are invisible containers that hold other Views (or other ViewGroups) and define
 * their layout properties.

大致意思是(翻译可能不标准,大家相互理解):View这个类代表了用户界面组件的基本构建块。一个视图在屏幕上占据一个矩形区域,并负责绘图和事件处理。视图是窗口小部件的基类,它是用于创建交互式的用户界面组件(按钮,文本等)。这类布局的基类,它是看不见的容器持有其他Views(或其他viewgroup)和定义他们的layout的属性。

其次为什么要理解View的绘制原理?个人认为只有理解View的绘制原理,才能学会如何去自定义一个View。我们知道开发中需要用到很多组件,很多时候android给出的组建并不能满足我们的开发需求,于是我们开始组合组件来使用,然而,发现有些特定的需求还是得不到满足,于是自定义View成为解决的首要期望。

接下来我上个特别简单效果,给TextView加个边框

android 如何知道每个view的绘制时间 android view的绘制原理_自定义


需要说明的是,这个例子没什么实际意义,只是一种假设

看一下代码

public class MyTextView extends TextView {

    protected Paint mPaint = new Paint();

    public MyTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);

    }

    public MyTextView(Context context, AttributeSet attrs) {
        super(context, attrs);

    }

    public MyTextView(Context context) {
        super(context);

    }

    @Override
    protected void onDraw(Canvas canvas) {
        canvas.save();
        mPaint.setColor(Color.RED);
        //文本
        String text = getText().toString();
        //文本宽度
        int textWidth = (int) mPaint.measureText(text);
        //文本高度
        int textHeight = (int) (mPaint.descent() - mPaint.ascent());
        //drawLine(float startX, float startY, float stopX, float stopY, Paint paint)
        //绘制矩形的上边
        canvas.drawLine(0, 0, textWidth, 0, mPaint);
        //绘制矩形的左边
        canvas.drawLine(0, 0, 0, textHeight, mPaint);
        //绘制矩形的下边
        canvas.drawLine(0, textHeight, textWidth, textHeight, mPaint);
        //绘制矩形的右边
        canvas.drawLine(textWidth, 0, textWidth, textHeight, mPaint);
        mPaint.setColor(Color.BLACK);
        //文本开始的高,相当于baseLine
        int y= textHeight+(int)(mPaint.descent() + mPaint.ascent()/2);
        //绘制文本
        canvas.drawText(text, 0, baseLine, mPaint);
        //保存绘制的状态,一定不能忘记
        canvas.restore();
    }

xml引用

<com.example.testview.MyTextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/hello_world" />

说了那么多,好像没提到View的绘制原理,其实已经说了,当然,并没有着重去说,接下来我谈谈自己的理解:
View的绘制主要是三部曲:
onMeasure:测量过程,计算视图的大小,通过setMeasuredDimension保存计算结果。
onLayout:保存位置
onDraw:绘制过程,可以绘制背景,视图本身等
开发中,我们必须去实现onMeasure,onDraw这两个方法。可能有人会问刚刚的MyTextView只实现了onDraw方法呀,额,貌似是的。但是,你要看清楚,MyTextView继承的是View的子类TextView,而不是直接继承View,TextView已经实现onMeasure方法

@Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        int width;
        int height;
 ...
        setMeasuredDimension(width, height);
    }

最后我认为还需要理解的有MeasureSpec和Canvas (这个类大家应该都掌握)这两个类
MeasureSpec这个类定义了有三种测量模式,EXACTLY(精确),AT_MOST(可获取最大尺寸),UNSPECIFIED(不限制)。这个类在onMeasure中使用。

在面试中我们经常遇到这个问题,我们可以说一下什么是View,然后着重讲一下View绘制的过程,开发中自己如何完成自定View的,再来个总结,这样面试官应该觉得至少自己掌握自定义View。我觉得照自己理解的去说给面试官听就好。。关于View的绘制原理,以后我会剖析源码,有时间再更新。最后,有问题的地方还请大家指出。