invalidate和requestLayout

Invalidate:
To farce a view to draw,call invalidate().——摘自View类源码
从上面这句话看出,invalidate方法会执行draw过程,重绘View树。
当View的appearance发生改变,比如状态改变(enable,focus),背景改变,隐显改变等,这些都属于appearance范畴,都会引起invalidate操作。

所以当我们改变了View的appearance,需要更新界面显示,就可以直接调用invalidate方法。

View(非容器类)调用invalidate方法只会重绘自身,ViewGroup调用则会重绘整个View树。

RequestLayout:
To initiate a layout, call requestLayout(). This method is typically called by a view on itself when it believes that it can no longer fit within its current bounds.——摘自View源码

从上面这句话看出,当View的边界,也可以理解为View的宽高,发生了变化,不再适合现在的区域,可以调用requestLayout方法重新对View布局。

View执行requestLayout方法,会向上递归到顶级父View中,再执行这个顶级父View的requestLayout,所以其他View的onMeasure,onLayout也可能会被调用。

总结:

View绘制分三个步骤,顺序是:onMeasure,onLayout,onDraw。经代码亲测,log输出显示:调用invalidate方法只会执行onDraw方法;调用requestLayout方法只会执行onMeasure方法和onLayout方法,并不会执行onDraw方法。

所以当我们进行View更新时,若仅View的显示内容发生改变且新显示内容不影响View的大小、位置,则只需调用invalidate方法;若View宽高、位置发生改变且显示内容不变,只需调用requestLayout方法;若两者均发生改变,则需调用两者,按照View的绘制流程,推荐先调用requestLayout方法再调用invalidate方法。

requestLayout示例:实现可移动组件:https://blog.csdn.net/qq_39658819/article/details/78994308

import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.FrameLayout;
 
/**
 * 步骤一:自定义可移动组件
 * @author NewBies
 * @date 2017/12/26
 */
public class VertexView extends android.support.v7.widget.AppCompatTextView{
 
    private int startX;
    private int startY;
    private int endX;
    private int endY;
    private FrameLayout.LayoutParams layoutParams;
 
    public VertexView(Context context) {
        super(context);
    }
 
    public VertexView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }
 
    public VertexView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }
 
 
    /**
     * 步骤二:重写onTouchEvent事件
     * @param event
     * @return
     */
    @Override
    public boolean onTouchEvent(MotionEvent event){
        //步骤三:获取手机触摸点的横坐标和纵坐标
        endX = (int)event.getX();
        endY = (int)event.getY();
 
        //步骤四:获取布局参数实例,注意:xx.LayoutParams这里的xx应该是该组件的父布局类型
        //注意:这句话必须在该组件已经添加到父布局中才会起作用,所以这句话我没有写在构造函数中,而是写在这里
        layoutParams = (FrameLayout.LayoutParams) this.getLayoutParams();
 
        switch (event.getAction()){
            //监听按下去的事件,这个事件在每次拖动时,必定会执行,也只执行一次
            case MotionEvent.ACTION_DOWN:
                //将按下去的点记录为起始点
                startX = endX;
                startY = endY;
                break;
            //步骤五:监听移动事件,该事件会在拖动时执行N次
            case MotionEvent.ACTION_MOVE:
                //计算移动的距离
                int offsetX = endX - startX;
                int offsetY = endY - startY;
 
                //调用layout方法来重新放置它的位置
                layoutParams.setMargins(getLeft() + offsetX, getTop() + offsetY, getRight() + offsetX, getBottom() + offsetY);
                //刷新
                requestLayout();
                break;
            //监听抬起事件,该事件同按下去的时间一样,只执行一次
            case MotionEvent.ACTION_UP:
                break;
            default:break;
        }
        //这里应该返回true,这里涉及到了android的事件拦截机制,大致意思是,我的事件是在哪里处理,就在那里的事件返回TRUE
        return  true;
    }
}

猜你喜欢

转载自www.cnblogs.com/genggeng/p/10014121.html