android广播

概念

  • 现实:电台通过发送广播发布消息,买个收音机,就能收听
  • Android:系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件。
    Android系统在运行的过程中,会产生很多事件,比如开机、电量改变、收发短信、拨打电话、屏幕解锁

广播接收者

  • 当一条广播被发送出来时,系统是在所有清单文件中遍历,通过匹配意图过滤器找到能接收这条广播的广播接收者

IP拨号器

原理:接收拨打电话的广播,修改广播内携带的电话号码

  • 定义广播接收者接收打电话广播
public class CallReceiver extends BroadcastReceiver {
    //当广播接收者接收到广播时,此方法会调用
    @Override
    public void onReceive(Context context, Intent intent) {
        //拿到用户拨打的号码
        String number = getResultData();
        //修改广播内的号码
        setResultData("17951" + number);
    }
}
  • 在清单文件中定义该广播接收者接收的广播类型
<receiver android:name="com.xxx.ipdialer.CallReceiver">
    <intent-filter >
        <action android:name="android.intent.action.NEW_OUTGOING_CALL"/>
    </intent-filter>
</receiver>
  • 接收打电话广播需要权限
<uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS"/>
  • 即使广播接收者的进程没有启动,当系统发送的广播可以被该接收者接收时,系统会自动启动该接收者所在的进程

短信拦截器

系统收到短信时会产生一条广播,广播中包含了短信的号码和内容

  • 定义广播接收者接收短信广播
public class SmsReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        //取出短信内容
        Bundle bundle = intent.getExtras();
        //数组中的每一个元素,就是一条短信
        Object[] objects = (Object[]) bundle.get("pdus");

        for (Object object : objects) {
            //把数组中的元素转换成短信对象
            SmsMessage sms = SmsMessage.createFromPdu((byte[]) object);
            //获取对方号码
            String address = sms.getOriginatingAddress();
            //获取短信内容
            String body = sms.getMessageBody();
            System.out.println(address + ";" + body);
            if("13888".equals(address)){
                //拦截短信
                abortBroadcast();
            }
        }
    }
}
  • 系统创建广播时,把短信存放到一个数组,然后把数据以pdus为key存入bundle,再把bundle存入intent
  • 清单文件中配置广播接收者接收的广播类型,注意要设置优先级属性,要保证优先级高于短信应用,才可以实现拦截
<receiver android:name="com.xxx.smslistener.SmsReceiver">
    <intent-filter android:priority="1000">//系统最大值
        <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
    </intent-filter>
</receiver>
  • 添加权限
<uses-permission android:name="android.permission.RECEIVE_SMS"/>
  • 4.0之后,广播接收者所在的应用必须启动过一次,才能生效
  • 4.0之后,如果广播接收者所在应用被用户手动关闭了,那么再也不会启动了,直到用户再次手动启动该应用

监听SD卡状态

  • 清单文件中定义广播接收者接收的类型,监听SD卡常见的三种状态,所以广播接收者需要接收三种广播
<receiver android:name="com.xxx.sdcradlistener.SDCardReceiver">
    <intent-filter >
        <action android:name="android.intent.action.MEDIA_MOUNTED"/>
        <action android:name="android.intent.action.MEDIA_UNMOUNTED"/>
        <action android:name="android.intent.action.MEDIA_REMOVED"/>
        <data android:scheme="file"/>
    </intent-filter>
</receiver>
  • 广播接收者的定义
public class SDReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        //判断收到的是神马广播
        //获取广播中的action
        String action = intent.getAction();
        if(Intent.ACTION_MEDIA_MOUNTED.equals(action)){
            Toast.makeText(context, "sd卡就绪", 0).show();
        }
        else if(Intent.ACTION_MEDIA_REMOVED.equals(action)){
            Toast.makeText(context, "sd卡被拔出了", 0).show();
        }
        else if(Intent.ACTION_MEDIA_UNMOUNTED.equals(action)){
            Toast.makeText(context, "sd卡被卸载了", 0).show();
        }
    }
}

监听应用的安装、卸载、更新

原理:应用在安装卸载更新时,系统会发送广播,广播里会携带应用的包名

  • 清单文件定义广播接收者接收的类型,因为要监听应用的三个动作,所以需要接收三种广播
<receiver android:name="com.xxx.app.AppReceiver">
    <intent-filter >
        <action android:name="android.intent.action.PACKAGE_ADDED"/>
        <action android:name="android.intent.action.PACKAGE_REPLACED"/>
        <action android:name="android.intent.action.PACKAGE_REMOVED"/>
        <data android:scheme="package"/>
    </intent-filter>
</receiver>
  • 广播接收者的定义
public void onReceive(Context context, Intent intent) {
    //区分接收到的是哪种广播
    String action = intent.getAction();
    //获取广播中包含的应用包名
    Uri uri = intent.getData();
    if(action.equals("android.intent.action.PACKAGE_ADDED")){
        System.out.println(uri + "被安装了");
    }
    else if(action.equals("android.intent.action.PACKAGE_REPLACED")){
        System.out.println(uri + "被更新了");
    }
    else if(action.equals("android.intent.action.PACKAGE_REMOVED")){
        System.out.println(uri + "被卸载了");
    }
}

发送自定义广播

public void click(View v){
    Intent intent = new Intent();
    intent.setAction("a.b.c");
    //发送自定义无序广播
    sendBroadcast(intent);
}

//有序广播
public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    public void click(View v){
        Intent intent = new Intent();
        intent.setAction("com.center.fdm");
        //发送自定义有序广播
        //resultReceiver:在所有广播接收者都收到广播之后,才会收到,一定是最后一个收到,并且一定能收到
        sendOrderedBroadcast(intent, null, new MyReceiver(), null, 0, "每人发100斤大米", null);
    }
    class MyReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(Context context, Intent intent) {
            String order = getResultData();
            System.out.println("反贪局收到文件:" + order);
        }
    }
}
public class ShengZF extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        //获取中央的红头文件
        String order = getResultData();
        System.out.println("省政府收到文件:" + order);
        //无序广播不能修改广播内容
        setResultData("每人发80斤大米");
    }
}
public class ShiZF extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        //获取中央的红头文件
        String order = getResultData();
        System.out.println("市政府收到文件:" + order);
        //无序广播不能拦截
        abortBroadcast();
    }
}
public class XianZF extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        //获取中央的红头文件
        String order = getResultData();
        System.out.println("县政府收到文件:" + order);
    }
}

广播的两种类型

  • 无序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,并且是没有先后顺序(同时收到)
  • 有序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,但是会按照广播接收者的优先级来决定接收的先后顺序
    • 优先级的定义:-1000~1000
    • 结果接收者:所有广播接收者都接收到广播之后,它才接收,并且一定会接收
    • abortBroadCast:阻止其他接收者接收这条广播,类似拦截,只有有序广播可以被拦截

进程优先级

  1. 前台进程:拥有一个正在与用户交互的activity(onResume方法被调用)的进程
  2. 可见进程:拥有一个非前台,但是对用户可见的activity(onPause方法被调用)的进程
  3. 服务进程:拥有一个通过startService方法启动的服务的进程
  4. 后台进程:拥有一个后台activity(onStop方法被调用)的进程
  5. 空进程:没有拥有任何活动的应用组件的进程,也就是没有任何服务和activity在运行

电话窃听器

  • 电话状态:空闲、响铃、接听
  • 获取电话管理器,设置侦听
TelephonyManager tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);
tm.listen(new MyPhoneStateListener(), PhoneStateListener.LISTEN_CALL_STATE);
  • 侦听对象的实现
class MyPhoneStateListener extends PhoneStateListener{

    //当电话状态改变时,此方法调用
    @Override
    public void onCallStateChanged(int state, String incomingNumber) {
        // TODO Auto-generated method stub
        super.onCallStateChanged(state, incomingNumber);
        switch (state) {
        case TelephonyManager.CALL_STATE_IDLE://空闲
            if(recorder != null){
                recorder.stop();
                recorder.release();
            }
            break;
        case TelephonyManager.CALL_STATE_OFFHOOK://摘机
            if(recorder != null){
                recorder.start();
            }
            break;
        case TelephonyManager.CALL_STATE_RINGING://响铃
            recorder = new MediaRecorder();
            //设置声音来源
            recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
            //设置音频文件格式
            recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
            recorder.setOutputFile("sdcard/haha.3gp");
            //设置音频文件编码
            recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
            try {
                recorder.prepare();
            } catch (IllegalStateException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            break;
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值