当前位置: 首页 > news >正文

完整教程:invalidate(),postInvalidate()和requestLayout()区别

完整教程:invalidate(),postInvalidate()和requestLayout()区别

这三个方法都是用于触发视图更新的,但它们的应用场景和触发的“更新级别”完全不同。

  • invalidate(): “重绘”。意思是“我当前的内容变了(比如颜色、文字、位置等),需要重新画一遍”。它只触发 onDraw() 方法。
  • postInvalidate(): “在非UI线程中安全地重绘”。功能和 invalidate() 完全一样,但它可以在非UI线程(子线程)中调用。
  • requestLayout(): “重新测量和布局”。意思是“我的尺寸可能变了,或者子视图的结构变了,整个布局需要重新计算”。它会触发完整的 measure() -> layout() 流程,可能也会触发onDraw()

详细对比

特性invalidate()postInvalidate()requestLayout()
核心作用标记视图的局部区域为脏区,请求重绘非UI线程中安全地请求重绘请求重新布局整个视图树。
触发方法onDraw(Canvas)onDraw(Canvas)onMeasure(), onLayout() (以及可能的 onDraw())
调用线程必须UI主线程中调用。可以在任何线程(主线程或子线程)中调用。必须UI主线程中调用。
性能开销较小。只影响自身视图的绘制。较小。同 invalidate()较大。会从根视图开始,可能遍历整个视图树,重新测量和布局所有相关视图。
使用场景内容改变但视图的大小和位置不变时。
例如:
- 改变背景色、文字颜色
- 在 onTouchEvent 中移动一个子视图
- 自定义View时动态改变绘制内容
在子线程中更新UI,例如:
- 在 AsyncTaskdoInBackground 中更新进度
- 在子线程中进行计算,并实时反馈到UI上
视图的边界(尺寸)可能发生变化时。
例如:
- 给 TextView 设置新的文字,导致其宽高改变
- 动态添加或移除子View
- 在自定义View中改变了 LayoutParams

深入解析与示例

1. invalidate()

当你只改变了视图的内容,而它的尺寸和位置没有变化时,使用 invalidate()

工作流程
invalidate() -> dispatchDraw() -> onDraw()

示例

public class CustomView extends View {
private int mCircleColor = Color.RED;
// 在UI线程中改变颜色并重绘
public void changeColor() {
mCircleColor = Color.BLUE;
invalidate(); // 触发onDraw,视图会变成蓝色
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Paint paint = new Paint();
paint.setColor(mCircleColor);
canvas.drawCircle(100, 100, 50, paint);
}
}
2. postInvalidate()

invalidate() 的线程安全版本。其内部实现是向主线程的Handler发送了一个消息,最终在主线程中调用了 invalidate()

示例

public class CustomView extends View {
private int mProgress = 0;
// 在子线程中更新进度
public void startUpdateInBackground() {
new Thread(new Runnable() {
@Override
public void run() {
while (mProgress < 100) {
mProgress++;
// 不能在子线程直接调用invalidate(),否则会崩溃
// invalidate(); // 错误!
postInvalidate(); // 正确!安全地在主线程触发重绘
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
// 绘制一个根据mProgress变化的进度条...
}
}
3. requestLayout()

当你认为当前视图的尺寸或布局已经不再满足要求,需要重新计算时,调用此方法。它会从ViewRootImpl开始,执行一个完整的遍历(Traversal)。
工作流程
requestLayout() -> onMeasure() -> onLayout() -> (可能) onDraw()

为什么可能触发 onDraw()
因为重新布局后,视图的位置和大小可能发生了变化,系统认为你需要重新绘制以适应新的布局。

示例

public class MyTextView extends TextView {
public void setTextAndResize(String text) {
setText(text);
// 设置新文字后,这个TextView所需的宽度和高度可能变了。
// 我们需要告诉父布局:“我的尺寸可能变了,请重新测量和摆放我”。
requestLayout();
}
// 或者在自定义View中,你重写了onMeasure,并且条件发生了变化
private boolean mIsWideMode = false;
public void setWideMode(boolean isWide) {
if (mIsWideMode != isWide) {
mIsWideMode = isWide;
// 测量逻辑改变了,必须请求重新布局
requestLayout();
}
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
if (mIsWideMode) {
// 宽模式的测量逻辑
setMeasuredDimension(500, 100);
} else {
// 窄模式的测量逻辑
setMeasuredDimension(200, 100);
}
}
}

如何选择?

  1. 只涉及视觉表现变化(颜色、位置、透明度等)?

    • -> 使用 invalidate()
    • 如果在子线程中 -> 使用 postInvalidate()
  2. 视图的尺寸或布局结构发生了变化(宽高、边距、子视图数量等)?

    • -> 使用 requestLayout()
  3. 不确定该用哪个?

    • 先想想你的改变是否影响了视图的尺寸。如果影响了,用 requestLayout();如果没影响,只用 invalidate()。滥用 requestLayout() 会导致不必要的性能损耗。

组合使用

在某些复杂情况下,你甚至可能需要同时调用两者。

// 例如,一个自定义View,它既改变了内部状态(需要重绘),又改变了自己的尺寸(需要重新布局)
public void complexChange() {
changeInternalState(); // 改变状态
requestLayout(); // 请求重新布局(这会隐式包含重绘)
// 或者,如果你确定invalidate()是必要的,也可以显式调用,但通常requestLayout()就够了。
// invalidate(); 
}

总结一下,理解这三个方法的区别,关键在于理解Android视图系统的工作流程:测量(Measure) -> 布局(Layout) -> 绘制(Draw)requestLayout() 触发了前两步(可能包括第三步),而 invalidate()/postInvalidate() 只触发了第三步。

http://www.zskr.cn/news/81821.html

相关文章:

  • 二手房翻新不踩坑!苏州本土 3 家口碑公司帮你实现老房逆袭(附 2025 避坑指南) - 品牌测评鉴赏家
  • 揭秘!这些整装服务强到逆天,新房装修闭眼选 - 品牌测评鉴赏家
  • 装修公司大揭秘:售后服务哪家强? - 品牌测评鉴赏家
  • 2025年12月成都电商小程序开发,预订服务小程序开发,活动报名小程序开发公司推荐:看综合实力 - 品牌鉴赏师
  • 算法第四次作业
  • 二手房翻新怎么选?这3类靠谱公司帮你避坑(附2025口碑榜单) - 品牌测评鉴赏家
  • re入门
  • 能工智人
  • 第三天—C++语法基础
  • 2025年12月超级充电桩,欧标充电桩,日标充电桩厂家推荐:行业权威盘点与品质红榜发布​ - 品牌鉴赏师
  • 2025新房整装服务哪家强?这份避坑指南+口碑榜单请收好 - 品牌测评鉴赏家
  • DSU on array - 反向操作区间合并
  • 关于Visual Studio 2022 Git无法使用的解决办法
  • Python 面向对象编程 (OOP) 核心:类、封装与继承
  • 12/10
  • 完整教程:分享一个基于服务端地图服务裁剪的方法
  • Nginx安全配置
  • 并发编程的三大基石:从底层逻辑聊透“同步、互斥与分工”
  • 在 .Net 8 WEBAPI 中实现实体框架的 Code First 办法
  • Coppersmith 学习笔记
  • 【SQL技术】不同数据库引擎 SQL 优化方案剖析 - 详解
  • Python list all files in dir recursivelly
  • python —— 树的遍历 —— 深度优先遍历(先序、中序、后序)
  • 恰好被k个区间覆盖的点的数量
  • windriver 第5章:USB 概述
  • Airflow - from airflow import DAG and from airflow.sdk import DAG, which one is better?
  • 货代邮件自动化处理系统设计文档
  • DSU on array
  • 吐血整理!新房全包装修,性价比之王大盘点 - 品牌测评鉴赏家
  • Resources资源同步加载、异步加载、卸载