Android之LocalBroadCastManager实现原理

LocalBroadcastManager实现原理

1:概述

我们都知道广播可以用来实现跨进程通信,它的原理就是binder,通过system_server来实现的,那么当我们只是在应用内部来实现通信的时候,其实我们可以考虑使用LocalBroadcastManager来实现,它有两个好处
1.1:性能较一般的broadcast要好,因为它的实现不需要经过binder调用,它采用的是handler通信来实现的
1.2:安全性较好,不必担心数据泄露或是被截取
LocalBroadcastManager是单利模式,并且只能通过代码的方式注册,不能通过xml的方式注册,下面是谷歌对其的解释
/**

  • Helper to register for and send broadcasts of Intents to local objects
  • within your process.
  • This has a number of advantages over sending
  • global broadcasts with {@link android.content.Context#sendBroadcast}:
  • You know that the data you are broadcasting won't leave your app, so
  • don't need to worry about leaking private data.
  • It is not possible for other applications to send these broadcasts to
  • your app, so you don't need to worry about having security holes they can
  • exploit.
  • It is more efficient than sending a global broadcast through the
  • system.
    */

2:实现原理

2.1:单例模式

@NonNull
public static LocalBroadcastManager getInstance(@NonNull Context context) {
    synchronized (mLock) { //锁
        if (mInstance == null) {//创建对象
            mInstance = new LocalBroadcastManager(context.getApplicationContext());
        }
        return mInstance;
    }
}

private LocalBroadcastManager(Context context) {
    mAppContext = context;
    //getMainLooper是主线程的对应的handler对象
    mHandler = new Handler(context.getMainLooper()) {

        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case MSG_EXEC_PENDING_BROADCASTS:
                    executePendingBroadcasts();
                    break;
                default:
                    super.handleMessage(msg);
            }
        }
    };
}

2.2注册广播registerReceiver

public void registerReceiver(@NonNull BroadcastReceiver receiver,
        @NonNull IntentFilter filter) {
    synchronized (mReceivers) {
        //创建一个ReceiverRecord对象
        ReceiverRecord entry = new ReceiverRecord(filter, receiver);
        //先从缓存的HashMap中查询,key为receiver
        ArrayList filters = mReceivers.get(receiver);
        if (filters == null) {
            filters = new ArrayList<>(1);
            mReceivers.put(receiver, filters);//缓存到hashmap列表中去
        }
        filters.add(entry);
        for (int i=0; i entries = mActions.get(action);
            if (entries == null) {
                entries = new ArrayList(1);
                mActions.put(action, entries);//缓存到action的Hashmap列表中
            }
            entries.add(entry);
        }
    }
}

这个过程主要是将数据缓存到两个列表中去

1: private final HashMap> mReceivers
= new HashMap<>(); key为receiver,value为ReceiverRecord列表

2: private final HashMap> mActions = new HashMap<>();
将action缓存到Hashmap中去,记录的是action与广播接收者之间的关系

2.3:取消注册unregisterReceiver

public void unregisterReceiver(@NonNull BroadcastReceiver receiver) {
    synchronized (mReceivers) {
        //从ArratList中移除广播接收者,remove方法返回的是value值
        final ArrayList filters = mReceivers.remove(receiver);
        if (filters == null) {//如果为null的话不做后续处理
            return;
        }
        for (int i=filters.size()-1; i>=0; i--) {
            final ReceiverRecord filter = filters.get(i);
            filter.dead = true;
            for (int j=0; j receivers = mActions.get(action);
                if (receivers != null) {
                    for (int k=receivers.size()-1; k>=0; k--) {
                        final ReceiverRecord rec = receivers.get(k);
                        if (rec.receiver == receiver) {
                            rec.dead = true;
                            receivers.remove(k);//从mActions移除接收者为空的action
                        }
                    }
                    if (receivers.size() <= 0) {
                        mActions.remove(action);
                    }
                }
            }
        }
    }
}

2.4发送广播sendBroadcast

public boolean sendBroadcast(@NonNull Intent intent) {
    synchronized (mReceivers) {
        final String action = intent.getAction();
        final String type = intent.resolveTypeIfNeeded(
                mAppContext.getContentResolver());
        final Uri data = intent.getData();
        final String scheme = intent.getScheme();
        final Set categories = intent.getCategories();

        final boolean debug = DEBUG ||
                ((intent.getFlags() & Intent.FLAG_DEBUG_LOG_RESOLUTION) != 0);
        if (debug) Log.v(
                TAG, "Resolving type " + type + " scheme " + scheme
                + " of intent " + intent);
        //intent读取action,根据action获取相应的广播接收者列表
        ArrayList entries = mActions.get(intent.getAction());
        if (entries != null) {
            if (debug) Log.v(TAG, "Action list: " + entries);

            ArrayList receivers = null;
            for (int i=0; i= 0) {
                    if (debug) Log.v(TAG, "  Filter matched!  match=0x" +
                            Integer.toHexString(match));
                    if (receivers == null) {
                        receivers = new ArrayList();
                    }
                    receivers.add(receiver);
                    receiver.broadcasting = true;
                } else {
                    if (debug) {
                        String reason;
                        switch (match) {
                            case IntentFilter.NO_MATCH_ACTION: reason = "action"; break;
                            case IntentFilter.NO_MATCH_CATEGORY: reason = "category"; break;
                            case IntentFilter.NO_MATCH_DATA: reason = "data"; break;
                            case IntentFilter.NO_MATCH_TYPE: reason = "type"; break;
                            default: reason = "unknown reason"; break;
                        }
                        Log.v(TAG, "  Filter did not match: " + reason);
                    }
                }
            }

            if (receivers != null) {
                for (int i=0; i

2.5:主线程消息队列:消息的处理

mHandler = new Handler(context.getMainLooper()) {

        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case MSG_EXEC_PENDING_BROADCASTS:
                    executePendingBroadcasts();//广播的处理
                    break;
                default:
                    super.handleMessage(msg);
            }
        }
    };

private void executePendingBroadcasts() {
    while (true) {
        final BroadcastRecord[] brs;
        synchronized (mReceivers) {
            final int N = mPendingBroadcasts.size();
            if (N <= 0) {
                return;
            }
            brs = new BroadcastRecord[N];//缓存到数组中,广播接收者可能有多个
            mPendingBroadcasts.toArray(brs);
            mPendingBroadcasts.clear();
        }
        for (int i=0; i

3:概括

优点:
1:相对于串行广播,更加安全,数据只会在应用内部传输,并且限定在同一进程中,一个APK中其他的进程也是不行的
2:性能较好:相对于串行广播,其不对system_server造成任何的性能问题,并且也没有跨进程调用带来的耗时问题
缺点:
1:不支持跨进程,如果跨进程必须考虑其他实现方式
2:不支持静态xml的方式注册,无法实现监听诸如系统开机广播等

你可能感兴趣的:(Android之LocalBroadCastManager实现原理)