Android实训案例(六)——四大组件之一BroadcastReceiver的基本使用,拨号,短信,SD卡,开机,应用安装卸载监听

简介: <div class="markdown_views"><h1 id="android实训案例六四大组件之一broadcastreceiver的基本使用拨号短信sd卡开机应用安装卸载监听">Android实训案例(六)——四大组件之一BroadcastReceiver的基本使用,拨号,短信,SD卡,开机,应用安装卸载监听</h1><hr><blockquote> <

Android实训案例(六)——四大组件之一BroadcastReceiver的基本使用,拨号,短信,SD卡,开机,应用安装卸载监听


Android中四大组件的使用时重中之重,我这个阶段也不奢望能把他所有的原理搞懂,但是最起码的,我要把他的各种使用方法了如指掌才行

BroadcastReceiver

接收系统的广播,比如电话,短信之类的

1.IP拨号器

我们在拨打电话的时候,我们系统也会事先发送一个广播,所以我们可以用广播接收者来接收到这个广播拨打电话的时候在电话号码前面加上一些优惠的长途短号,其逻辑就是入门使用广播的一个小案例,那我们新建一个IPCall项目

这里写图片描述

我们拨打电话的时候系统发一个广播,我们接受到这个广播拿到号码修改之后再放回去,就达到了我们IP拨号的目的,我们需要新建一个Receiver

CallReceiver

package com.lgl.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

/**
 * 电话广播
 * Created by lgl on 16/4/10.
 */
public class CallReceiver extends BroadcastReceiver {

    //接收到广播时调用
    @Override
    public void onReceive(Context context, Intent intent) {
        Log.i("CallReceiver", "打电话");

    }
}

这个类暂时不需要做什么。最重要的还是注册,作为四大组建,他是需要在清单文件里注册的

 <!--注册广播-->
        <receiver android:name=".CallReceiver">
            <intent-filter>
                <!--定义接收的广播-->
                <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
            </intent-filter>
        </receiver>

别忘了,你监听了用户的隐私,是需要权限的

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

我们先运行一下,打个电话

这里写图片描述

好的,广播我们已经抓住了,那我们可以首先获取打出去的号码

String iphone = getResultData();
Log.i("CallReceiver", "电话号码:"+iphone);

这样我们就可以拿到String类型的电话号码了

这里写图片描述

我们现在可以修改号码重新设置IP段了

        //添加IP段
        String newPhone = "+86" + iphone;
        //把修改后的号码放回去
        setResultData(newPhone);

这样我们每次打电话他都会自动帮我们添加一个字段了

这里写图片描述

完整代码

package com.lgl.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

/**
 * 电话广播
 * Created by lgl on 16/4/10.
 */
public class CallReceiver extends BroadcastReceiver {

    //接收到广播时调用
    @Override
    public void onReceive(Context context, Intent intent) {
        Log.i("CallReceiver", "打电话");
        String iphone = getResultData();
        Log.i("CallReceiver", "电话号码:" + iphone);
        //添加IP段
        String newPhone = "+86" + iphone;
        //把修改后的号码放回去
        setResultData(newPhone);

    }
}

2.短信拦截器

系统受到了一条短信后,也是会发一条广播的,所有我们可以在中间写一个广播接受者去进行我们的操作,这里,我们继续在IP拨号器这个项目中写吧,就不新建项目了,不然等下上传也麻烦

我们新建一个Class——SMSReceiver,重写他的onReceive方法,然后我们先注册

<!--注册广播-->
        <receiver android:name=".SMSReceiver">
            <intent-filter>
                <!--定义接收的广播,被Google隐藏的权限操作-->
                <action android:name="android.provider.Telephony.SMS_RECEIVED" />
            </intent-filter>
        </receiver>

切記,权限

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

SMSReceiver

package com.lgl.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.telephony.SmsMessage;
import android.util.Log;

/**
 * 短信拦截器
 * Created by lgl on 16/4/10.
 */
public class SMSReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        System.out.println("收到短信");
        //获取短信内容
        Bundle bundle = intent.getExtras();
        //返回的是一个Object数组
        Object[] objects = (Object[]) bundle.get("pdus");
        //遍历数组得到短信内容
        for (Object object : objects) {
            //把数组元素转换成短信对象
            SmsMessage sms = SmsMessage.createFromPdu((byte[]) object);
            //获取发件人号码
            String toPhone = sms.getOriginatingAddress();
            //获取短信内容
            String smsContent = sms.getMessageBody();
            Log.i("SMSReceiver", "发件人号码:" + toPhone + "短信内容" + smsContent);

            //判断是否是拦截的号码
            if (toPhone.equals("12345678910")) {
                //拦截广播
                abortBroadcast();
            }
        }
    }
}

这样我们运行之下,你会发现,虽然走到拦截这一步,但是并没有阻止显示在短信收件箱里,这里,我们要注意一个优势,就是广播接收者是有优先级定义的,我们只需要在清单注册根节点的intent-filter标签里定义一个

android:priority="1000"

官方文档是说数值在-1000到1000之间,但是最高支持int的最大值的权限,int最大值是多少?自己去看API

3.监听SD卡

我们在对SD卡进行读写的时候会用到,其实也就是巩固一下对广播的使用,做那种语音助手之类的辅助软件,广播和服务还是很有用的,我们还是定义一个SDReceiver并且在清单文件注册

 <!--注册广播-->
        <receiver android:name=".SDReceiver">
            <intent-filter>
                <!--SD卡就绪广播-->
                <action android:name="android.intent.action.MEDIA_MOUNTED" />
                <!--SD卡拔出广播-->
                <action android:name="android.intent.action.MEDIA_REMOVED" />
                <!--SD卡卸载广播-->
                <action android:name="android.intent.action.MEDIA_UNMOUNTABLE" />
                <data android:scheme="file"/>
            </intent-filter>
        </receiver>

我们现在可以监听了

SDReceiver

package com.lgl.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

/**
 * 监听SD卡
 * Created by lgl on 16/4/10.
 */
public class SDReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        //判断广播
        String action = intent.getAction();
        if (action.equals(Intent.ACTION_MEDIA_MOUNTED)) {
            Log.i("SDReceiver", "SD卡就绪");
        } else if (action.equals(Intent.ACTION_MEDIA_REMOVED)) {
            Log.i("SDReceiver", "SD卡拔出");
        } else if (action.equals(Intent.ACTION_MEDIA_UNMOUNTABLE)) {
            Log.i("SDReceiver", "SD卡卸载");
        }
    }
}

4.流氓软件

我们监听到开机就启动这个软件,而且不让其退出,达到流氓的效果

这里写图片描述

先不让他退出,我们在MainActivity中

    @Override
    public void onBackPressed() {
        //禁止返回键
        // super.onBackPressed();
    }

只要你一安装,就退出不了了,我们再设置一下开机启动,写一个监听器动的广播罢了,我们新建一个RebootReceiver,先注册吧

<!--注册广播-->
        <receiver android:name=".RebootReceiver">
            <intent-filter>
                <!--定义接收的广播-->
                <action android:name="android.intent.action.BOOT_COMPLETED" />
            </intent-filter>
        </receiver>

这个也是需要权限的

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

RebootReceiver

package com.lgl.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

/**
 * 开机启动
 * Created by lgl on 16/4/10.
 */
public class RebootReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Log.i("RebootReceiver", "开机");
        //启动
        Intent i = new Intent(context, MainActivity.class);
        //在Activity之外启动需要设置Flags
        i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(i);
    }
}

5.应用安装卸载监听

我们如果做手机助手或者应用的时候或许可以用得上这玩意,新建一个APPReceiver,然后去注册

<!--注册广播-->
        <receiver android:name=".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>

然后我们来判断

package com.lgl.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.util.Log;

/**
 * Created by lgl on 16/4/10.
 */
public class APPReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        //判断广播类型
        String action = intent.getAction();
        //获取包名
        Uri appName = intent.getData();

        if (Intent.ACTION_PACKAGE_ADDED.equals(action)) {
            Log.i("APPReceiver", "安装" + appName);
        } else if (Intent.ACTION_PACKAGE_REPLACED.equals(action)) {
            Log.i("APPReceiver", "更新" + appName);
        } else if (Intent.ACTION_PACKAGE_REMOVED.equals(action)) {
            Log.i("APPReceiver", "卸载" + appName);
        }
    }
}

这样我们完全就可以监听状态了,我们以下载一个豌豆荚为例

这里写图片描述

6.自定义广播

这里,我们有特定需求的时候会用,我们先定义一个Button

         <Button
            android:id="@+id/btn_myreceiver"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="发送自定义广播" />

他的点击事件就是发送一个广播,我这里自定义一个广播名字liuguilin

             case R.id.btn_myreceiver:
                Intent i = new Intent();
                i.setAction("liuguilin");
                sendBroadcast(i);
                break;

然后我们新建一个MyReceiver,注册


        <receiver android:name=".MyReceiver">
            <intent-filter>
                <!--自定义广播-->
                <action android:name="liuguilin" />
            </intent-filter>
        </receiver>

我们写个打印语句

这里写图片描述

7.有序广播和无序广播

这两种广播的区别

  • 有序广播:接收这条广播是按优先级来的
  • 无序广播:无条件直接接收

-1.发送有序广播

我们定义一个Button

         <Button
            android:id="@+id/btn_haveorder"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="发送有序广播" />

然后发送一个广播

case R.id.btn_haveorder:
                Intent intent = new Intent();
                intent.setAction("com.lgl.good");
                //有序广播并且携带数据
                sendOrderedBroadcast(intent, null, null, null, 0, "自定义广播内容", null);
                break;

这里我定义了三个广播,然后他们的优先级分辨是1000,600,300


        <!--有序广播-->
        <receiver android:name=".OrderReceiver">
            <intent-filter android:priority="1000">
                <action android:name="com.lgl.good"/>
            </intent-filter>
        </receiver>
        <!--有序广播-->
        <receiver android:name=".OrderReceiverTwo">
            <intent-filter android:priority="600">
                <action android:name="com.lgl.good"/>
            </intent-filter>
        </receiver>

        <!--有序广播-->
        <receiver android:name=".OrderReceiverThree">
            <intent-filter android:priority="300">
                <action android:name="com.lgl.good"/>
            </intent-filter>
        </receiver>

最后运行的结果

这里写图片描述

Demo下载:http://download.csdn.net/detail/qq_26787115/9486705

目录
相关文章
|
1天前
|
缓存 移动开发 Android开发
构建高效Android应用:从优化用户体验到提升性能表现
【4月更文挑战第18天】 在移动开发的世界中,打造一个既快速又流畅的Android应用并非易事。本文深入探讨了如何通过一系列创新的技术策略来提升应用性能和用户体验。我们将从用户界面(UI)设计的简约性原则出发,探索响应式布局和Material Design的实践,再深入剖析后台任务处理、内存管理和电池寿命优化的技巧。此外,文中还将讨论最新的Android Jetpack组件如何帮助开发者更高效地构建高质量的应用。此内容不仅适合经验丰富的开发者深化理解,也适合初学者构建起对Android高效开发的基础认识。
2 0
|
1天前
|
移动开发 Android开发 开发者
构建高效Android应用:采用Kotlin进行内存优化的策略
【4月更文挑战第18天】 在移动开发领域,性能优化一直是开发者关注的焦点。特别是对于Android应用而言,由于设备和版本的多样性,确保应用流畅运行且占用资源少是一大挑战。本文将探讨使用Kotlin语言开发Android应用时,如何通过内存优化来提升应用性能。我们将从减少不必要的对象创建、合理使用数据结构、避免内存泄漏等方面入手,提供实用的代码示例和最佳实践,帮助开发者构建更加高效的Android应用。
4 0
|
3天前
|
缓存 移动开发 Java
构建高效的Android应用:内存优化策略
【4月更文挑战第16天】 在移动开发领域,尤其是针对资源有限的Android设备,内存优化是提升应用性能和用户体验的关键因素。本文将深入探讨Android应用的内存管理机制,分析常见的内存泄漏问题,并提出一系列实用的内存优化技巧。通过这些策略的实施,开发者可以显著减少应用的内存占用,避免不必要的后台服务,以及提高垃圾回收效率,从而延长设备的电池寿命并确保应用的流畅运行。
|
5天前
|
搜索推荐 开发工具 Android开发
安卓即时应用(Instant Apps)开发指南
【4月更文挑战第14天】Android Instant Apps让用户体验部分应用功能而无需完整下载。开发者需将应用拆分成模块,基于已上线的基础应用构建。使用Android Studio的Instant Apps Feature Library定义模块特性,优化代码与资源以减小模块大小,同步管理即时应用和基础应用的版本。经过测试,可发布至Google Play Console,提升用户便利性,创造新获客机会。
|
5天前
|
Java API 调度
安卓多线程和并发处理:提高应用效率
【4月更文挑战第13天】本文探讨了安卓应用中多线程和并发处理的优化方法,包括使用Thread、AsyncTask、Loader、IntentService、JobScheduler、WorkManager以及线程池。此外,还介绍了RxJava和Kotlin协程作为异步编程工具。理解并恰当运用这些技术能提升应用效率,避免UI卡顿,确保良好用户体验。随着安卓技术发展,更高级的异步处理工具将助力开发者构建高性能应用。
|
5天前
|
编解码 人工智能 测试技术
安卓适配性策略:确保应用在不同设备上的兼容性
【4月更文挑战第13天】本文探讨了提升安卓应用兼容性的策略,包括理解平台碎片化、设计响应式UI(使用dp单位,考虑横竖屏)、利用Android SDK的兼容工具(支持库、资源限定符)、编写兼容性代码(运行时权限、设备特性检查)以及优化性能以适应低端设备。适配性是安卓开发的关键,通过这些方法可确保应用在多样化设备上提供一致体验。未来,自动化测试和AI将助力应对设备碎片化挑战。
|
11天前
|
移动开发 API Android开发
构建高效Android应用:探究Kotlin协程的优势与实践
【4月更文挑战第7天】 在移动开发领域,性能优化和应用响应性的提升一直是开发者追求的目标。近年来,Kotlin语言因其简洁性和功能性在Android社区中受到青睐,特别是其对协程(Coroutines)的支持,为编写异步代码和处理并发任务提供了一种更加优雅的解决方案。本文将探讨Kotlin协程在Android开发中的应用,揭示其在提高应用性能和简化代码结构方面的潜在优势,并展示如何在实际项目中实现和优化协程。