Android进阶_Handler和Handler.Callback和Message

最近得病了,看到代码黄色警报不是很爽。虽然不影响运行,但是看着代码不干净不利索。


Handler,线程通信,用的比较多吧,今天写代码一如往常的:

	private Handler smsHandler = new Handler(){
		public void handleMessage(Message msg) {
			
		};
	};
感觉没什么错误,怎么出黄色警报了。

This Handler class should be static or leaks might occur 

大体意思是说 Handler应该是静态的,否则会造成内存溢出。

然后将Handler用static修饰,ok,警报消失。


但是遇到问题了:不设置成静态会造成内存溢出?要毁三观?然后就开始寻求网络资源的帮助

http://www.cnblogs.com/jevan/p/3168828.html

在这里找到了答案,MessageQueue中的消息队列会一直持有对handler的引用,而作为内部类的handler会一直持有外部类的引用,就会导致外部类不能被GC回收。当我们发延时很长的msg时就容易出现泄漏。所以此处应该设置为static,然后Handler就会跟随类而不是跟随类的对象加载,也就不再持有外部类的对象引用。

单单这个问题可以直接看上面这篇文章。


通过搜索还找到了另一个解决黄色警报的解决方案:在新建Handler对象的时候传入一个Handler.Callback对象。

	private Handler mHandler = new Handler(new Handler.Callback() {

		@Override
		public boolean handleMessage(Message msg) {
			return false;
		}
	});
以前还没注意过这个参数呢,这个方法,返回值都是干啥的?

然后去看了一下这一块的源码:

    /**
     * Callback interface you can use when instantiating a Handler to avoid
     * having to implement your own subclass of Handler.
     *
     * @param msg A {@link android.os.Message Message} object
     * @return True if no further handling is desired
     */
    public interface Callback {	// callback接口
        public boolean handleMessage(Message msg);
    }
    
    /**
     * Subclasses must implement this to receive messages.
     */
    public void handleMessage(Message msg) {
    }
    
    /**
     * Handle system messages here.
     */
    public void dispatchMessage(Message msg) {
        if (msg.callback != null) { // callback是msg中的一个字段,是一个Runnable对象,当通过handler.post方法发送一个runnable的时候就会被封装到这个msg中
            handleCallback(msg); // 此方法是Handler中的一个静态方法,方法体:message.callback.run();只有这一句,可以看出是直接调用的run()方法,没有新建线程,否则也不符合这里线程通信了对吧。
        } else {
            if (mCallback != null) { // mCallback就是上面接口的对象
                if (mCallback.handleMessage(msg)) { // 如果返回true直接return结束方法,不再调用handler中的handleMessage方法。
                    return;
                }
            }
            handleMessage(msg);// handler中的消息处理方法
        }
    }

Looper从MQ中拿到一个msg就会调用handler中的dispatchMessage方法,然后有此方法进行分发处理。想到触摸时间分发了木有?

最后记录一下Message对象的获取,原来一直使用Message.obtain方法来获取,今天无意间发现了handler.obtainMessage方法也可以获取。

通过源码发现,其实handler.obtainMessage最终也是调用了Message中的一系列obtain方法。



你可能感兴趣的:(Android进阶)