Handler线程通信机制。

Handler在Android中起到至关重要的作用, 主要体现在线程通信, 来更新UI界面。

1.使用方式

package com.example.alldemo;

import android.app.Activity;
import android.app.ActionBar;
import android.app.Fragment;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.os.Build;

public class MainActivity extends Activity {
	
	private Handler handler = new Handler(){

		@Override
		public void handleMessage(Message msg) {
			super.handleMessage(msg);
			//这里更新界面
		}
		
	};

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		new Thread(new Runnable() {
			
			@Override
			public void run() {
				Message message = Message.obtain();   //这里获取已经存在的消息, 不要自己实例化。
				handler.sendMessage(message);
			}
		});
	}

}


2. 原理

package android.os;

import android.util.Log;
import android.util.Printer;
import android.util.PrefixPrinter;


public class Looper {
    private static final String TAG = "Looper";

    // sThreadLocal.get() will return null unless you've called prepare().
    static final ThreadLocal<Looper> sThreadLocal = new ThreadLocal<Looper>();

    final MessageQueue mQueue;
    final Thread mThread;
    volatile boolean mRun;

    private Printer mLogging = null;
    private static Looper mMainLooper = null;  // guarded by Looper.class

     /** 
      * 给当前线程设置一个局部变量Looper。每个线程只能设置一个Looper
      */
    public static void prepare() {
        if (sThreadLocal.get() != null) {
            throw new RuntimeException("Only one Looper may be created per thread");
        }
        sThreadLocal.set(new Looper());
    }

    public static void prepareMainLooper() {
        prepare();
        setMainLooper(myLooper());
        myLooper().mQueue.mQuitAllowed = false;
    }

    private Looper() {
        mQueue = new MessageQueue();
        mRun = true;
        mThread = Thread.currentThread();
    }

}

 1.首先使用handle需要在当前线程中执行Looper.prepare(),作用是给当前线程初始化消息队列。

    那这里有个疑问, 为什么在UI线程中为什么不先初始化就能使用, 这个是因为在底层源码中系统已经调用了。

   下面是ActivityThread的main方法中的代码

public static final void main(String[] args) {
        SamplingProfilerIntegration.start();

        Process.setArgV0("<pre-initialized>");

        Looper.prepareMainLooper();  //这里执行的prepare
        if (sMainThreadHandler == null) {
            sMainThreadHandler = new Handler();
        }

        ActivityThread thread = new ActivityThread();
        thread.attach(false);

        if (false) {
            Looper.myLooper().setMessageLogging(new
                    LogPrinter(Log.DEBUG, "ActivityThread"));
        }

        Looper.loop(); //这里loop

        if (Process.supportsProcesses()) {
            throw new RuntimeException("Main thread loop unexpectedly exited");
        }

        thread.detach();
        String name = (thread.mInitialApplication != null)
            ? thread.mInitialApplication.getPackageName()
            : "<unknown>";
        Slog.i(TAG, "Main thread of " + name + " is now exiting");
    }
}

2. 就是发送消息,说的更明白点就是将消息加入消息队列里。

public boolean sendMessageAtTime(Message msg, long uptimeMillis)
    {
        boolean sent = false;
        MessageQueue queue = mQueue;
        if (queue != null) {
            msg.target = this;  //msg.target 就是handler自己
            sent = queue.enqueueMessage(msg, uptimeMillis);
        }
        else {
            RuntimeException e = new RuntimeException(
                this + " sendMessageAtTime() called with no mQueue");
            Log.w("Looper", e.getMessage(), e);
        }
        return sent;
    }


这一块是加入消息队列以后按时间排序

final boolean enqueueMessage(Message msg, long when) {
        if (msg.isInUse()) {
            throw new AndroidRuntimeException(msg
                    + " This message is already in use.");
        }
        if (msg.target == null && !mQuitAllowed) {
            throw new RuntimeException("Main thread not allowed to quit");
        }
        final boolean needWake;
        synchronized (this) {
            if (mQuiting) {
                RuntimeException e = new RuntimeException(
                    msg.target + " sending message to a Handler on a dead thread");
                Log.w("MessageQueue", e.getMessage(), e);
                return false;
            } else if (msg.target == null) {
                mQuiting = true;
            }

            msg.when = when;
            //Log.d("MessageQueue", "Enqueing: " + msg);
            Message p = mMessages;
            if (p == null || when == 0 || when < p.when) {
                msg.next = p;
                mMessages = msg;
                needWake = mBlocked; // new head, might need to wake up
            } else {
                Message prev = null;
                while (p != null && p.when <= when) {
                    prev = p;
                    p = p.next;
                }
                msg.next = prev.next;
                prev.next = msg;
                needWake = false; // still waiting on head, no need to wake up
            }
        }
        if (needWake) {
            nativeWake(mPtr);
        }
        return true;
    }


3. 最后一步就是Looper.loop() 。为什么ui线程中没有,请见上面的源码。

public static void loop() {
        Looper me = myLooper();
        if (me == null) {
            throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");
        }
        MessageQueue queue = me.mQueue;
        
        // Make sure the identity of this thread is that of the local process,
        // and keep track of what that identity token actually is.
        Binder.clearCallingIdentity();
        final long ident = Binder.clearCallingIdentity();
        
        while (true) {
            Message msg = queue.next(); // might block
            if (msg != null) {
                if (msg.target == null) {
                    // No target is a magic identifier for the quit message.
                    return;
                }

                long wallStart = 0;
                long threadStart = 0;

                // This must be in a local variable, in case a UI event sets the logger
                Printer logging = me.mLogging;
                if (logging != null) {
                    logging.println(">>>>> Dispatching to " + msg.target + " " +
                            msg.callback + ": " + msg.what);
                    wallStart = SystemClock.currentTimeMicro();
                    threadStart = SystemClock.currentThreadTimeMicro();
                }

                msg.target.dispatchMessage(msg);

                if (logging != null) {
                    long wallTime = SystemClock.currentTimeMicro() - wallStart;
                    long threadTime = SystemClock.currentThreadTimeMicro() - threadStart;

                    logging.println("<<<<< Finished to " + msg.target + " " + msg.callback);
                    if (logging instanceof Profiler) {
                        ((Profiler) logging).profile(msg, wallStart, wallTime,
                                threadStart, threadTime);
                    }
                }

                // Make sure that during the course of dispatching the
                // identity of the thread wasn't corrupted.
                final long newIdent = Binder.clearCallingIdentity();
                if (ident != newIdent) {
                    Log.wtf(TAG, "Thread identity changed from 0x"
                            + Long.toHexString(ident) + " to 0x"
                            + Long.toHexString(newIdent) + " while dispatching to "
                            + msg.target.getClass().getName() + " "
                            + msg.callback + " what=" + msg.what);
                }
                
                msg.recycle();//这里回收消息, 所以使用的时候使用回收的就行。
            }
        }
    }

循环依次取出Message, 然后执行 msg.target.dispatchMessage(msg); msg.targer 就是handler。

/**
     * Handle system messages here.
     */
    public void dispatchMessage(Message msg) {
        if (msg.callback != null) {
            handleCallback(msg);
        } else {
            if (mCallback != null) {
                if (mCallback.handleMessage(msg)) {
                    return;
                }
            }
            handleMessage(msg);
        }
    }

最后是调用了handleMessage()。








  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Handler线程通信机制是指在多线程环境下,通过Handler进行线程间的通信和消息传递的机制HandlerAndroid中的一个类,它主要用于异步消息的处理和线程通信。当一个Handler被创建时,它会自动和当前线程的Looper对象进行绑定,从而使得该Handler可以接收和处理与该Looper关联的消息队列中的消息。 在线程间通信中,一般会涉及到两个关键的角色:发送者和接收者。发送者通过Handler将消息封装成Message对象,然后调用Handler的sendMessage()方法将消息发送到消息队列中。而接收者则通过Handler的handleMessage()方法来处理接收到的消息。 具体的线程通信流程如下: 1. 发送者线程创建一个Handler对象,并与自己的Looper进行绑定。 2. 发送者线程创建一个Message对象,并设置消息内容。 3. 发送者线程调用Handler的sendMessage()方法,将消息发送到消息队列中。 4. 接收者线程中的Looper会不断地从消息队列中取出消息,并交给接收者线程Handler的handleMessage()方法进行处理。 5. 接收者线程Handler处理完消息后,可以通过sendMessage()方法将结果发送给发送者线程或其他线程。 通过Handler线程通信机制,可以实现不同线程间的通信和协调。比如在子线程中进行耗时操作,然后将结果通过Handler回传到主线程进行UI更新等操作。 ### 回答2: 在Android开发中,Handler线程通信机制一种用于在不同线程间进行通信机制。它可以将消息和任务发送到目标线程的消息队列中,然后在目标线程中执行这些消息和任务。 1. Handler的创建和使用: 首先,我们需要在目标线程中创建一个Handler对象。可以在主线程中创建Handler对象,并传入一个Looper对象作为参数。然后,我们可以通过调用Handler对象的post()、sendMessage()或其他类似方法,将消息或任务发送到目标线程的消息队列中。 2. 消息处理: 在目标线程中,Looper对象会不断地从消息队列中获取消息,然后根据消息的类型和内容进行处理。可以通过重写Handler类的handleMessage()方法来定义如何处理特定类型的消息。当有新消息到达时,Looper会调用这个方法来处理消息。 3. 线程通信: 通过使用Handler线程通信机制,我们可以在不同线程之间传递消息和任务。例如,我们可以在后台线程中执行耗时的计算,并使用Handler将计算结果发送到主线程,然后在主线程中更新UI界面。这样,我们就可以实现后台计算和UI更新的异步处理。 4. 消息队列和消息循环: Handler线程通信机制基于消息队列和消息循环的概念。消息队列是一个先进先出的数据结构,用于存储消息和任务。消息循环是一个不断循环的过程,在每次循环中,消息队列会被检查,如果队列中有消息,就会被取出并处理。 5. 线程安全: Handler线程通信机制线程安全的,因为消息队列的访问是同步的。不同线程可以通过Handler对象将消息和任务发送到同一个消息队列中,这些消息和任务会被逐个处理,避免了多线程并发访问的问题。 总之,Handler线程通信机制Android开发中一种常用的线程间通信机制,它可以将消息和任务发送到目标线程的消息队列中,然后在目标线程中处理这些消息和任务。它在后台计算和UI更新、线程安全等方面都具有重要的作用。 ### 回答3: Handler线程通信机制Android一种常见的线程间通信方式。HandlerAndroid中的一个类,它功能强大,可以用于创建一个与特定线程关联的消息队列,从而实现线程间的通信。 在使用Handler线程通信机制时,一般分为两个角色:发送消息的线程和接收消息的线程。发送消息的线程使用Handler对象发送消息,而接收消息的线程则通过重载Handler类的handleMessage()方法来处理接收到的消息。发送消息的线程将消息封装成Message对象,然后通过Handler的sendMessage()方法将消息发送给接收消息的线程。接收消息的线程在handleMessage()方法中根据消息的种类来处理相应的任务。 Handler线程通信机制实现主要有以下几个步骤: 1. 创建Handler对象:在接收消息的线程中创建一个Handler对象,用于接收和处理消息。 2. 发送消息:在发送消息的线程中,通过Handler对象的sendMessage()方法将消息发送给接收消息的线程。消息可以通过Message类来创建,并且可以通过Message的arg1、arg2和obj属性来传递一些额外的数据。 3. 处理消息:接收消息的线程需要重载Handler类的handleMessage()方法,该方法用于处理接收到的消息。根据消息的种类,可以使用switch语句来处理不同的消息类型。 4. 更新UI:如果接收消息的线程和UI线程需要进行通信,可以在handleMessage()方法中更新UI相关的内容。可以使用post()方法或者runOnUiThread()方法来在UI线程中更新UI。 5. 销毁Handler:在不需要使用Handler对象的时候,应该调用Handler的removeCallbacksAndMessages()方法取消消息的发送,并且释放相关资源。 通过Handler线程通信机制,可以实现不同线程之间的通信和协作,提高应用程序的灵活性和响应性。它是Android开发中常用的一种线程间通信方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值