handler.post(r)同一个线程的疑惑

Handler.post(Runnable r)将任务加入消息队列,并不开启新线程,而是等待主线程消息循环处理。程序启动时,setContentView先执行,界面加载,但内容显示在Activity的Resume阶段。LogCat先打印信息,10秒后TextView文字在界面显示,可通过检查界面元素ID验证这一过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

handler.post(r);是把r加到消息队列,但并未开辟新线程。等到消息被取出时才执行。

package com.lei.handlethread;

import android.os.Bundle;
import android.os.Handler;
import android.app.Activity;
import android.view.Menu;
import android.widget.Button;

public class MainActivity extends Activity {
    private Button btn = null;
    private Handler handler = new Handler();
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        handler.post(r);
        setContentView(R.layout.activity_main);
        btn = (Button)findViewById(R.id.hello);// 用来验证setContentView()先执行的。      
        String s=(String) btn.getText();// 
        System.out.println(s);
        System.out.println("activity--->"+Thread.currentThread().getId());
        System.out.println("Activityname--->"+Thread.currentThread().getName());
    }
    
    Runnable  r = new Runnable() {
         public void run() {
                System.out.println("handler--->"+Thread.currentThread().getId());
                System.out.println("handlername--->"+Thread.currentThread().getName());
                try {
                    Thread.sleep(10000);
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
         }
        
    };
   
        

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }
}

运行结果:logCat先打印如下信息。程序运行界面过10s显示TextView文字。

解释:

main线程从消息泵中取出一个消息,处理(执行相关函数),然后再取一个,处理。所以onCreate是某一消息处理中的执行,其中post一个消息,只是把消息加入队列了,还没执行新消息,什么时候执行?要等前一个消息处理完,再次从消息泵中取消息处理时,它才被执行。

所以先是main的system.out,再是post的system.out

相比之下,sendMessage是同步执行,用handler.sendMessage,那顺序就变了。

至于setContentView(R.layout.activity_main);肯定是最先执行,程序界面最先打开了,但是界面空间要等到Activity的Resume(即交互阶段)阶段才会显示。通过获取界面空间ID,在Log中打印空间内容就可验证。


### Handler.post 的使用方法 `Handler.post()` 是 `Handler` 提供的一个重要功能,用于将任务提交到与其关联的消息队列中执行。此方法允许开发者在指定线程的上下文中运行一段代码片段。 #### 方法签名 以下是 `post` 方法的主要形式: ```java public final boolean post(Runnable r) ``` 该方法接受一个实现了 `Runnable` 接口的对象作为参数,并将其放入消息队列中等待执行[^5]。 当调用 `handler.post(r)` 时,`r.run()` 将会在与 `Handler` 关联的线程上被执行。如果未显式绑定其他 Looper,则默认会是在主线程上执行。 #### 示例代码 下面是一个简单的示例展示如何利用 `Handler.post()` 来更新 UI: ```java // 创建一个基于主线程Looper的Handler实例 Handler handler = new Handler(Looper.getMainLooper()); // 定义一个匿名内部实现 Runnable接口的任务 Runnable task = new Runnable() { @Override public void run() { // 更新UI组件的操作放在这里,比如修改TextView的内容 TextView textView = findViewById(R.id.textView); textView.setText("Updated by Handler!"); } }; // 把这个任务加入到Handler管理的消息队列里去异步执行 if (!handler.hasMessages(0)) { handler.post(task); } ``` 上述代码展示了怎样通过 `Handler.post()` 调度一个更改界面控件状态的动作至主线程环境下来完成安全地刷新视图的工作流程[^6]。 另外还存在延时版本的方法可以设定延迟时间后再执行相应指令: ```java long delayMillis = 2000; // 延迟两秒 handler.postDelayed(new Runnable(){ @Override public void run(){} },delayMillis ); ``` 以上就是关于 Android 中 `Handler.post` 的基本介绍及其典型应用场景之一——跨线程间传递数据并最终影响到图形用户界面上的表现方式[^7]。 ### 注意事项 尽管 `Handler.post()` 非常方便易用,但在某些特殊场景下需要注意避免内存泄漏等问题的发生。例如长时间持有对外部对象(如 Activity 或 Fragment)强引用可能导致这些本应被回收销毁掉的东西无法释放资源从而造成潜在风险。因此建议尽可能采用弱引用模式或者及时清除回调函数等方式加以防范[^8]。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值