Android 中handler消息机制的理解

博主前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住也分享一下给大家,
👉点击跳转到网站

前言

Android中在子线程中更改主线程中的组件是不允许的。
具体规则就是:子线程不允许操作主线程中的组件。

如果我们必须在子线程中更改主线程中的UI组件怎么办?
所以Android 就为我们提供了handler

Handler相关API如下:

1、Message:消息

1、可以理解为线程间通讯的数据单元,可通过Message携带需要的数据。
2、创建对象:Message.obtain(what)
3、封装数据
public int what //id标识
public int arg1
public int arg2
public Object obj

2、Handler:处理器

1、Handler是Message的处理器,同时也负责消息的发送和移除的工作。
2、发送即时消息:sendMessage(Message message)
3、发送延时消息:sendMessageDelayed(Message msg,long time)
这里所说的延时消息是指,消息发送后,延时处理.也是立即发送,只是延时处理了而已。
4、handlerMessage(Message msg) 回调方法
5、移除还未处理的消息:removeMessages(int what)
6、移除所有未处理的消息 removeCallbacksAndMessages(null)

3、MessageQueue:消息队列

1、用来存放通过Handler发送的消息
2、它是一个按Message的when排序的优先级队列。when就是指的时间,发送的如果是即时消息,那么when就是当前时间,如果发送的是延时消息,那么就是当前时间+延时的时间

4、Looper(钩子):循环器

1、负责循环取出MessageQueue里面的当前需要处理的Message
2、交给对应的Handler进行处理
3、处理完后,将Message缓存到消息池中,以备复用。

Handler消息机制的原理图一
在这里插入图片描述
Handler消息机制的原理图二
在这里插入图片描述

看下面的一个例子理解handler的:

在这里插入图片描述
具体代码:

public class HandleActivity extends AppCompatActivity implements View.OnClickListener {
    private TextView tv_text;
    private Button btn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_handle);
        tv_text = findViewById(R.id.tv_text);
        btn = findViewById(R.id.btn);
        btn.setOnClickListener(this);
    }

    Handler handler = new Handler(new Handler.Callback() {
        @Override
        public boolean handleMessage(@NonNull Message msg) {
            if (msg.what == 0x00) {
                tv_text.setText("我是第二条内容");
            }
            return false;
        }
    });

    @Override
    public void onClick(View v) {
        //创建子线程
        new Thread(new Runnable() {
            @Override
            public void run() {
                handler.sendEmptyMessage(0x00);
            }
        }).start();
    }
}

具体思路就是:在点击按钮时,发送一个消息给handler,再由handler更新UI组件
总结:Handler是Android中提供的一个消息处理的机制

  1. 在任意线程中发送消息
  2. 在主线程中获取并处理消息

通过handler消息机制,实现进度条的功能:
在这里插入图片描述
xml布局如下:

<ProgressBar
        android:id="@+id/progress_bar"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        style="@style/Widget.AppCompat.ProgressBar.Horizontal"
        android:max="60"
        />

Activity中代码:

public class HandleActivity extends AppCompatActivity{
	private ProgressBar progress_bar;
    private static final int TIME=60;//定义时间长度为60秒
    private int mProgressStatus=0; //定义完成的进度
    private final int TIMER_MSG=0X00; //声明消息代码
 @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_handle);
		//获取进度条组件
        progress_bar = findViewById(R.id.progress_bar);
        //启动进度条
        handler.sendEmptyMessage(TIMER_MSG);
}
	//创建handler对象,实现1秒钟更新一次进度
	Handler handler = new Handler(new Handler.Callback() {
        @Override
        public boolean handleMessage(@NonNull Message msg) {
            if (TIME-mProgressStatus>0){ //当前进度大于0
                mProgressStatus++; //进度加一
                progress_bar.setProgress(TIME-mProgressStatus);//更新进度
                handler.sendEmptyMessageDelayed(TIMER_MSG,1000);//一秒后发送消息
            }else {
                Toast.makeText(HandleActivity.this, "时间到啦,游戏结束!", Toast.LENGTH_SHORT).show();
            }
            return false;
        }
    });
}

Message语法:

  1. 使用Message.obtain或Handler.obtainMessage()方法获取message
  2. 携带int型信息,优先使用Message的arg1和arg2属性
  3. 使用Message.what来标记信息

Message对象的属性有:

  1. arg1,arg2整型
  2. obj Object类型
  3. replyTo 发送到何处
  4. what自定义的消息代码

Looper对象的创建:
handler在主线程中 是系统自动创建Looper对象
如果在子线程中创建Handler对象,则要先手动创建Looper对象。
代码如下:

public class LooperActivity extends AppCompatActivity {
    private static final String TAG = "LooperActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_looper);

        new Thread(new Runnable() {
            @Override
            public void run() {
                //初始化Looper对象
                Looper.prepare();
                Handler handler = new Handler(new Handler.Callback() {
                    @Override
                    public boolean handleMessage(@NonNull Message msg) {
                        Log.e(TAG, "handleMessage: " + String.valueOf(msg.what));
                        return false;
                    }
                });
                Message message = handler.obtainMessage(); //获取Message
                message.what = 0x4; //设置消息代码
                handler.sendMessage(message);
                //启动looper
                Looper.loop();
            }
        }).start();
    }
}

一、使用Handler实现异步工作

具体代码如下

public class GetWorkActivity extends AppCompatActivity {
    private Button btn_get;
    private TextView tv_content;
    private ProgressBar progressBar;
    private Button btn_handler_get;
    private Handler handler = new Handler(Looper.getMainLooper()) {
        @Override
        public void handleMessage(@NonNull Message msg) {//在主线程执行
            super.handleMessage(msg);
            //在handlerMessage中处理消息
            if (msg.what == 1) {
                String result = (String) msg.obj;
                tv_content.setText(result);
                progressBar.setVisibility(View.GONE);
            }
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_get_work);
        btn_get = findViewById(R.id.btn_get);
        tv_content = findViewById(R.id.tv_content);
        progressBar = findViewById(R.id.progressBar);
        btn_handler_get = findViewById(R.id.btn_handler_get);
        btn_get.setOnClickListener(new MyOnClick());
        btn_handler_get.setOnClickListener(new MyOnClick());
    }

    class MyOnClick implements View.OnClickListener {

        @Override
        public void onClick(View v) {
            switch (v.getId()) {
                case R.id.btn_get:
                    progressBar.setVisibility(View.VISIBLE);
                    GetOkHttp();
                    break;
                case R.id.btn_handler_get:
                    progressBar.setVisibility(View.VISIBLE);
                    GetHandlerOkHttp();
                    break;

            }
        }
    }


    private void GetOkHttp() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                OkHttpClient okHttpClient = new OkHttpClient();
                Request request = new Request.Builder()
                        .url("https://www.httpbin.org/get")
                        .build();
                try {
                    Response response = okHttpClient.newCall(request).execute();
                    String string = response.body().string();
                    runOnUiThread(new Runnable() {//子线程切换到主线程
                        @Override
                        public void run() {
                            tv_content.setText(string);
                            progressBar.setVisibility(View.GONE);
                        }
                    });
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }


    private void GetHandlerOkHttp() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                OkHttpClient okHttpClient = new OkHttpClient();
                Request request = new Request.Builder()
                        .url("https://www.httpbin.org/get")
                        .build();
                try {
                    Response response = okHttpClient.newCall(request).execute();
                    String string = response.body().string();
                    //在子线程创建Message对象
                    Message message = Message.obtain();
                    message.what = 1;//标识
                    message.obj = string;
                    //使用handler对象发送message
                    handler.sendMessage(message);
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 7
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

路宇

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值