关于android:Android入门教程-广播机制-Broadcast

40次阅读

共计 10695 个字符,预计需要花费 27 分钟才能阅读完成。

Android 利用能够通过播送从零碎或其余 App 接管或发送音讯。相似于订阅 - 公布设计模式。当某些事件产生时,能够收回播送。零碎在某些状态扭转时会收回播送,例如开机、充电。App 也可发送自定义播送。播送可用于利用间的通信,是 IPC 的一种形式。

播送的品种

播送的品种也能够看成是播送的属性。

  • 规范播送(Normal Broadcasts)
    齐全异步的播送。播送收回后,所有的播送接收器简直同时接管到这条播送。不同的 App 能够注册并接到规范播送。例如零碎播送。
  • 有序播送(Ordered Broadcasts)
    同步播送。同一时刻只有一个播送接收器能接管到这条播送。这个接收器解决完后,播送才会持续传递。有序播送是全局的播送。
  • 本地播送(Local Broaddcasts)
    只在本 App 发送和接管的播送。注册为本地播送的接收器无奈收到规范播送。
  • 带权限的播送
    发送播送时能够带上相干权限,申请了权限的 App 或播送接收器能力收到相应的带权限的播送。如果在 manifest 中申请了相应权限,接收器能够不必再申请一次权限即可接到相应播送。

接管播送

创立播送接收器,调用 onReceive() 办法,须要一个继承 BroadcastReceiver 的类。

注册播送
代码中注册称为动静注册。在 AndroidManifest.xml 中注册称为动态注册。动静注册的刚波接收器肯定要勾销注册。在 onDestroy() 办法中调用 unregisterReceiver() 办法来勾销注册。

不要在 onReceive() 办法中增加过多的逻辑操作或耗时的操作。因为在播送接收器中不容许开启线程,当 onReceive() 办法运行较长时间而没完结时,程序会报错。因而播送接收器个别用来关上其余组件,比方创立一条状态栏告诉或启动一个服务。

新建一个 MyExampleReceiver 继承自BroadcastReceiver

public class MyExampleReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {Toast.makeText(context,"Got it",Toast.LENGTH_SHORT).show();
        //abortBroadcast();}
}

abortBroadcast()能够截断有序播送

AndroidManifest.xml 中注册播送接收器;android:name里填接收器的名字。能够设置播送接收器优先级:

<intent-filter android:priority="100">

<receiver android:name=".MyExampleReceiver">
    <intent-filter>
        <action android:name="com.rust.broadcasttest.MY_BROADCAST"/>
    </intent-filter>
</receiver>

让接收器接管到一条 "com.rust.broadcasttest.MY_BROADCAST" 播送。
发送自定义播送(规范播送)时,要传送这个值。例如:

Intent intent = new Intent("com.rust.broadcasttest.MY_BROADCAST");
sendBroadcast(intent);

发送有序播送,该当调用sendOrderedBroadcast()

Intent intent = new Intent("com.rust.broadcasttest.MY_BROADCAST");
sendOrderedBroadcast(intent,null);

发送播送

App 有 3 种发送播送的形式。发送播送须要应用 Intent 类。

sendOrderedBroadcast(Intent, String)

发送有序播送。每次只有 1 个播送接收器能接到播送。接收器接到有序播送后,能够齐全地截断播送,或者传递一些信息给下一个接收器。有序播送的程序可受 android:priority 标签影响。同等级的接收器收到播送的程序是随机的。

sendBroadcast(Intent)

以一个未定义的程序向所有接收器发送播送。也称作一般播送。这种形式更高效,然而接收器不能给下一个接收器传递音讯。这类播送也无奈截断。

**LocalBroadcastManager.sendBroadcast
播送只能在应用程序外部进行传递,并且播送接收器也只能接管到来自本应用程序收回的播送。这个办法比全局播送更高效(不须要 Interprocess communication,IPC),而且不须要放心其它 App 会收到你的播送以及其余平安问题。

播送与权限

发送带着权限的播送

当你调用 sendBroadcast(Intent, String)sendOrderedBroadcast(Intent, String, BroadcastReceiver, Handler, int, String, Bundle)时,你能够指定一个权限。
接收器在 manifest 中申请了相应权限时能力收到这个播送。

例如发送一个带着权限的播送

sendBroadcast(new Intent("com.example.NOTIFY"), Manifest.permission.SEND_SMS);

接管播送的 app 必须注册相应的权限

<uses-permission android:name="android.permission.SEND_SMS"/>

当然也能够应用自定义。在 manifest 中应用 permission 标签

<permission android:name="custom_permission" />

增加后编译一下。即可调用Manifest.permission.custom_permission

接管带权限的播送

若注册播送接收器时申明了权限,那么只会接管到带着相应权限的播送。

在配置文件中申明权限,程序能力拜访一些要害信息。例如容许查问零碎网络状态。

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>

<!-- 机器开机播送 -->
<uses-permission android:name="android.permission.BOOT_COMPLETED">

如果没有申请权限,程序可能会意外敞开。

应用示例

发送和接管播送。分为发送和接管方 2 个 App。

应用带权限的播送。零碎权限与自定义权限。应用权限须要在 AndroidManifest.xml 中申明。如果是自定义权限,须要先增加自定义权限。

<!-- 自定义的权限  给播送用 -->
<permission android:name="com.rust.permission_rust_1" />
<uses-permission android:name="com.rust.permission_rust_1" />

发送播送时带上权限申明。接管方(不管是否己方 App)须要在 AndroidManifest.xml 中申请权限。注册接收器时也须要申明权限。

发送不带权限的有序播送

Intent intent = new Intent(MSG_PHONE);
sendOrderedBroadcast(intent, null);
Log.d(TAG, "[RustFisher-App1] 发送不带权限的有序播送," + intent.getAction());

发送方 App1 代码

private static final String TAG = "rustApp";
    public static final String MSG_PHONE = "msg_phone";
    public static final String PERMISSION_RUST_1 = "com.rust.permission_rust_1";

        // onCreate 注册播送接收器
        registerReceiver(mStandardReceiver1, makeIF());
        registerReceiver(mStandardReceiver2, makeIF());
        registerReceiver(mStandardReceiver3, makeIF());

        registerReceiver(mStandardReceiverWithPermission, makeIF(),
                Manifest.permission.permission_rust_1, null);  // 带上权限

        LocalBroadcastManager.getInstance(getApplicationContext())
                .registerReceiver(mLocalReceiver1, makeIF());
        LocalBroadcastManager.getInstance(getApplicationContext())
                .registerReceiver(mLocalReceiver2, makeIF());
        LocalBroadcastManager.getInstance(getApplicationContext())
                .registerReceiver(mLocalReceiver3, makeIF());

        // 解除接收器
        unregisterReceiver(mStandardReceiver1);
        unregisterReceiver(mStandardReceiver2);
        unregisterReceiver(mStandardReceiver3);

        unregisterReceiver(mStandardReceiverWithPermission);

        LocalBroadcastManager.getInstance(getApplicationContext())
                .unregisterReceiver(mLocalReceiver1);
        LocalBroadcastManager.getInstance(getApplicationContext())
                .unregisterReceiver(mLocalReceiver2);
        LocalBroadcastManager.getInstance(getApplicationContext())
                .unregisterReceiver(mLocalReceiver3);

    // 发送规范播送
    private void sendStandardBroadcast() {Intent intent = new Intent(MSG_PHONE);
        sendBroadcast(intent);
        Log.d(TAG, "[RustFisher-App1] Dispatcher 发送规范播送");
    }

    // 发送带权限的规范播送
    private void sendStandardBroadcastWithPermission() {Intent intent = new Intent(MSG_PHONE);
        sendBroadcast(intent, PERMISSION_RUST_1);
        Log.d(TAG, "[RustFisher-App1] Dispatcher 发送带权限的规范播送");
    }

    // 发送本地播送
    private void sendAppLocalBroadcast() {Intent intent = new Intent(MSG_PHONE);
        LocalBroadcastManager.getInstance(getApplicationContext()).sendBroadcast(intent);
        Log.d(TAG, "[RustFisher-App1] Dispatcher 发送本地播送");
    }

    private IntentFilter makeIF() {IntentFilter intentFilter = new IntentFilter(MSG_PHONE);
        intentFilter.addAction(Intent.ACTION_TIME_TICK);
        intentFilter.addAction(Intent.ACTION_TIME_CHANGED);
        return intentFilter;
    }

    // 规范接收器  用 context 来注册
    private BroadcastReceiver mStandardReceiver1 = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 规范接收器 1 收到:" + intent.getAction());
        }
    };

    // 规范接收器  用 context 来注册
    private BroadcastReceiver mStandardReceiver2 = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 规范接收器 2 收到:" + intent.getAction());
            if (intent.getAction().endsWith(MSG_PHONE)) {abortBroadcast(); // 截断有序播送
                Log.d(TAG, "[RustFisher-App1] 规范接收器 2 截断有序播送" + intent.getAction());
            }
        }
    };

    // 规范接收器  用 context 来注册
    private BroadcastReceiver mStandardReceiver3 = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 规范接收器 3 收到:" + intent.getAction());
        }
    };

    // 注册的时候给它带权限  规范接收器
    private BroadcastReceiver mStandardReceiverWithPermission = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 带权限的规范接收器收到:" + intent.getAction());
        }
    };

    /**
 * 用 LocalBroadcastManager 来注册成为本地接收器
 * 收不到规范播送 - 不论是本 app 收回的还是别的中央收回来的
 */
    private BroadcastReceiver mLocalReceiver1 = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 本地接收器 1 收到:" + intent.getAction());
        }
    };

    private BroadcastReceiver mLocalReceiver2 = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 本地接收器 2 收到:" + intent.getAction());
        }
    };

    private BroadcastReceiver mLocalReceiver3 = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[RustFisher-App1] 本地接收器 3 收到:" + intent.getAction());
        }
    };

接管方 App2 代码

<!-- 自定义的权限  给播送用 -->
<permission android:name="com.rust.permission_rust_1" />
<uses-permission android:name="com.rust.permission_rust_1" />
public static final String MSG_PHONE = "msg_phone";

    // onCreate 里注册接收器
    registerReceiver(mDefaultReceiver, makeIF());
    LocalBroadcastManager.getInstance(getApplicationContext())
            .registerReceiver(mLocalReceiver, makeIF());

    unregisterReceiver(mDefaultReceiver);
    LocalBroadcastManager.getInstance(getApplicationContext())
            .unregisterReceiver(mLocalReceiver);

    private BroadcastReceiver mDefaultReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[App2] standard receive:" + intent.getAction());
        }
    };

    private BroadcastReceiver mLocalReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {Log.d(TAG, "[App2] local receive:" + intent.getAction());
        }
    };

    private IntentFilter makeIF() {IntentFilter intentFilter = new IntentFilter(MSG_PHONE);
        intentFilter.addAction(Intent.ACTION_TIME_TICK);
        intentFilter.addAction(Intent.ACTION_TIME_CHANGED);
        return intentFilter;
    }

应用 LocalBroadcastManager 收回的本地播送,另一个 App 是接管不到的。要收到本地播送,同样须要 LocalBroadcastManager 来注册接收器。

能够把本地播送看成是一个部分的,App 内的播送体系。

试验中咱们留神到,Intent.ACTION_TIME_TICK播送是能够截断的。

监听屏幕亮灭

应用播送监听设施屏幕亮灭状态。这个是零碎收回来的播送。

应用的 action 是

  • Intent.ACTION_SCREEN_ON 亮屏
  • Intent.ACTION_SCREEN_OFF 灭屏

      private void registerScreenListener() {IntentFilter filter = new IntentFilter();
          filter.addAction(Intent.ACTION_SCREEN_ON);
          filter.addAction(Intent.ACTION_SCREEN_OFF);
          registerReceiver(mScreenReceiver, filter);
      }
    
      private BroadcastReceiver mScreenReceiver = new BroadcastReceiver() {
          @Override
          public void onReceive(Context context, Intent intent) {final String action = intent.getAction();
              if (Intent.ACTION_SCREEN_ON.equals(action)) {// 屏幕亮} else if (Intent.ACTION_SCREEN_OFF.equals(action)) {// 屏幕灭}
          }
      };

Broadcast 相干面试题

1. 播送传输的数据是否有限度,是多少,为什么要限度?

  • 播送是通过 Intent 携带须要传递的数据的
  • Intent 是通过 Binder 机制实现的
  • Binder 对数据大小有限度,不同 room 不一样,个别为 1M

2. 播送的分类?

  • 规范播送:通过 context. sendBroadcast 或者 context. sendBroadcastAsUser 发送给以后零碎中所有注册的接受者,也就是只有注册了就会接管到。利用在须要告诉各个播送接收者的状况下应用,如开机启动。
  • 有序播送:接收者依照优先级解决播送,并且后面解决播送的接受者能够停止播送的传递,个别通过 context. sendOrderedBroadcast 或者 context.sendOrderedBroadcastAsUser,在须要有特定拦挡的场景下应用,如黑名单短信、电话拦挡。
  • 粘性播送:能够发送给当前注册的接受者,意思是零碎会将后面的粘性播送保留在 AMS 中,一旦注册了与以保留的粘性播送合乎的播送,在注册完结后会立刻收到播送,个别通过 context. sendStickyBroadcast 或 context.sendStickyOrderedBroadcast 来发送,从字面上看,能够看进去粘性播送也分为一般粘性播送和有序粘性播送。
  • 本地播送:收回的播送只能在应用程序外部进行传递,播送接收器也只能承受来自本应用程序的播送。
  • 全局播送:零碎和播送,收回的播送能够被其余任何应用程序接管到,并且也能够承受到其余任何应用程序的播送。

3. 播送的应用场景,应用形式
播送是一种宽泛使用的在应用程序之间传输信息的机制,次要用来监听系统或者利用收回的播送信息,而后依据播送信息作为相应的逻辑解决,也能够用来传输大量、频率低的数据。

在实现开机启动服务和网络状态扭转、电量变动、短信和复电时通过接管零碎的播送让应用程序作出相应的解决。

应用:

// 在 AndroidManifest 中动态注册
<receiver
    android:name=".MyBroadcastReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter android:priority="100">
        <action android:name="com.example.hp.broadcasttest.MY_BROADCAST"/>
    </intent-filter>
</receiver>

// 动静注册,在代码中注册
@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    mIntentFilter = new IntentFilter();
    // 增加播送想要监听的类型,监听网络状态是否发生变化
    mIntentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
    mNetworkChangeReceiver = new NetworkChangeReceiver();
    // 注册播送
    registerReceiver(mNetworkChangeReceiver, mIntentFilter);
}

@Override
protected void onDestroy() {super.onDestroy();
    // 勾销注册播送接收器
    unregisterReceiver(mNetworkChangeReceiver);
}

// 发送播送,同样通过 Intent
Intent intent = new Intent("com.example.hp.broadcasttest.MY_BROADCAST");
// 发送规范播送
sendBroadcast(intent);

// 接管播送
public class MyBroadcastReceiver extends BroadcastReceiver {public MyBroadcastReceiver() { }

    @Override
    public void onReceive(Context context, Intent intent) {
        // TODO: This method is called when the BroadcastReceiver is receiving
        // an Intent broadcast.
        Toast.makeText(context, "received", Toast.LENGTH_SHORT).show();
        // 将这条播送截断
//        abortBroadcast();}
}

4. BroadcastReceiver,LocalBroadcastReceiver 区别
播送接收者:

(1)用于利用间的传递音讯(2)因为跨利用,存在平安问题

本地播送接收者:

(1)播送数据在本利用范畴内流传。(2)不必放心别的利用伪造播送。(3)比发送全局播送更高效、平安。(4)无奈应用动态注册  

5. 在 manifest 和代码中如何注册和应用 BroadcastReceiver
(1)在 AndroidManifest 中动态注册,而后间接应用。
(2)代码中,通过 registerReceiver 来注册。
(3)注册发送后,在 BroadcastReceiver(自定义一个接收器继承自 BroadcastReceiver)的 onReceive 中接管播送并解决播送。

6. 播送引起 anr 的工夫限度
前台播送:BROADCAST_FG_TIMEOUT = 10s
后盾播送:BROADCAST_BG_TIMEOUT = 60s

Android 入门教程视频参考

正文完
 0