android基础之四——广播与服务(一)

01_为什么需要广播接受者

1、电台:中央人民广播电台 93.4mhz;
2、收音机:买一个收音机,插上电池,调整到一个频道,接收广播;

Android系统中内置了广播电台(系统自带的广播功能),会广播出来一些事件(电量不足、电量
充满、接收到短信、外拨电话、拨进来电话、SD状态、开机启动等),指定了接收者广播消息的客
户端软件(自己开发的应用程序)可以做一些对用户有用的事情。

02广播接受者案例ip拨号器(重点)

在拨打电话的时候,把IP电话自动的添加到目标电话号码的前面,让拨打IP电话:
1、监听到用户外拨电话的事件:
2、得到用户外拨的电话号码,然后把ip电话添加目标电话号码的前面:
3、让用户拨打已经修改后电话号码:

写广播接收者的步骤:
1、买一个收音机:
    public class OutCallBroadcastReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {
        }
    }

2、插上电池:
     <receiver android:name="com.itheima.ipcall.OutCallBroadcastReceiver">
    </receiver>

3、调整到一个频道:
     <receiver android:name="com.itheima.ipcall.OutCallBroadcastReceiver">
        <intent-filter >
            <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
        </intent-filter>
    </receiver>

代码:

package com.itheima.ipcall;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
//1、监听到用户外拨电话的事件:
    public class OutCallBroadcastReceiver extends BroadcastReceiver {

        /**
         * 接收到广播消息后调用这个方法
         */
        @Override
        public void onReceive(Context context, Intent intent) {

            //2、得到用户外拨的电话号码,
            String phone = getResultData();
            SharedPreferences  sp = context.getSharedPreferences("info", Context.MODE_PRIVATE);
            String ipPhone  = sp.getString("ipPhone", "");

            //然后把ip电话添加目标电话号码的前面:
            phone = ipPhone + phone;

//          3、让用户拨打已经修改后电话号码:
            setResultData(phone);
        }

    }

配置文件:

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

      <receiver android:name="com.itheima.ipcall.OutCallBroadcastReceiver">
        <intent-filter >
            <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
        </intent-filter>
    </receiver>

03广播接受者案例短信监听器(重点)

pdus:protocol data unit s 协议数据单元;

广播接收者的特点:

1、即使广播接收者没有运行,当广播消息到达的时候系统会自动的启懂广播接收者,调用onReceive方法处理
消息;
2、在android4.0版本之后的系统,强制要求带广播接收者的应用程序必须有界面,并且至少运行过一次。
3、在android4.0版本之后的系统,强制要求带广播接收者的应用程序被冻结后,就不能运行了,只能在手工
启动的时候才有效。

代码: package com.itheima.smslistener;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.telephony.SmsMessage;

public class SMSBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {

        //得到短信数据
        Object[] objs = (Object[]) intent.getExtras().get("pdus");

        for(Object obj :objs){
            //obj是一条短信的byte数组
            SmsMessage sms = SmsMessage.createFromPdu((byte[])obj);
            //得到短信内容
            String content = sms.getMessageBody();
            //得到发送短信的电话号码
            String phone = sms.getOriginatingAddress();
            System.out.println("content==="+content);
            System.out.println("phone==="+phone);
        }
    }
}

配置文件:

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

    <receiver android:name="com.itheima.smslistener.SMSBroadcastReceiver">
            <intent-filter>
                <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
            </intent-filter>
    </receiver>

04广播接受者案例sd卡状态监听(重点)

测试的时使用4.0之前的模拟器或者真机,因为4.0之后的版本上已经去调用SD插拔、移除的功能;

代码:

package com.itheima.sdlistener;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

public class SDBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {

        String action = intent.getAction();
        if(action.equals("android.intent.action.MEDIA_MOUNTED")){
            Toast.makeText(context, "SD卡已经插上", 0).show();
        }
        if(action.equals("android.intent.action.MEDIA_UNMOUNTED")){
            Toast.makeText(context, "SD卡已经被拔掉", 0).show();
        }

        if(action.equals("android.intent.action.MEDIA_REMOVED")){
            Toast.makeText(context, "SD卡已经被移除", 0).show();
        }
    }

}

配置文件:

<receiver android:name="com.itheima.sdlistener.SDBroadcastReceiver">
        <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>

05广播接受者案例开机启动(重点)

1、让界面没法关闭,没法最小化;
2、如果支付成功,关闭当前界面;

代码:

MainActivity.java:

package com.itheima.bootlistener;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    public void pay(View view) {

        Toast.makeText(this, "已经支付了100元", 0).show();
        // 支付完成后关闭当前界面
        finish();
    }

    @Override
    public void onBackPressed() {
        // 按返回键不执行任何操作
    }

}

BootCompletedBroadcastReceiver.java:

package com.itheima.bootlistener;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.sax.StartElementListener;

public class BootCompletedBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {

        Intent i = new Intent(context,MainActivity.class); 
        //告诉Activity使用自己的任务栈来维护界面
        i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

        context.startActivity(i);
    }
}

配置文件:

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

  <receiver android:name="com.itheima.bootlistener.BootCompletedBroadcastReceiver">
        <intent-filter >
            <action android:name="android.intent.action.BOOT_COMPLETED"/>
        </intent-filter>
  </receiver>

06广播接受者案例卸载安装(重点)

代码:

package com.itheima.azxzlistener;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

public class AZXZBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction();

        if("android.intent.action.PACKAGE_INSTALL".equals(action)){
            Toast.makeText(context, "有一个新软件安装了......", 0).show();
        }
        if("android.intent.action.PACKAGE_REPLACED".equals(action)){
            Toast.makeText(context, "有一个软件升级了......", 0).show();
        }
        if("android.intent.action.PACKAGE_REMOVED".equals(action)){
            Toast.makeText(context, "有一个软件被卸载了......", 0).show();
        }

    }

}

配置文件:

      <receiver android:name="com.itheima.azxzlistener.AZXZBroadcastReceiver">
        <intent-filter >
            <action android:name="android.intent.action.PACKAGE_INSTALL"/>
            <action android:name="android.intent.action.PACKAGE_REMOVED"/>
            <action android:name="android.intent.action.PACKAGE_REPLACED"/>
          <!--   必须添加这个属性 -->
            <data android:scheme="package"/>
        </intent-filter>
    </receiver>

07_发送自定义广播

自定义广播的代码:

    Intent intent = new Intent();
    //设置广播的事件类型
    intent.setAction("com.itheima.broadcast.FSZDYGB");
    //设置广播传递的数据
    intent.putExtra("info", "自定义的广播消息");

    //发送广播消息
    sendBroadcast(intent);

08_有序广播和无序广播(重点)

有序广播:当广播消息发送出去之后,可以根据广播接收者的优先级的高低,从高到低一级一级的传递消息;
无序广播:当广播消息发送出去之后,只要是指定了广播事件类型的广播接收者都可以接受到这个消息;

发送有序的广播:

  • 自定义的有序广播:

    package com.itheima.broadcast;

    import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.view.View;

    public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    
    public void send(View view ){
    
        Intent intent = new Intent();
        // 设置广播的事件类型
        intent.setAction("com.itheima.broadcast.GWYFFNTBT");
    
        // 发送有序广播消息
        // 发送广播的意图对象
        // receiverPermission 指定广播接收者的权限,只有指定了这个权限的广播接收者才能接收到这个广播消息
        //resultReceiver 指定哪个广播接收者最后接收到广播消息
        //scheduler 消息处理器
        //initCode初始码
        //initData 初始数据
        //extraData 传递的额外参数
        sendOrderedBroadcast(intent, null, new ResultBroadcastReceiver(),
                null, 1, "发送2014年农田补贴,每亩地补贴900元", null);
    }
    

    }

  • 自定义的广播:

    代码:

    package com.itheima.gov;
    
    import android.content.BroadcastReceiver;
    import android.content.Context;
    import android.content.Intent;
    
    public class TownBroadcastReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            //得到广播的初始数据
            String data = getResultData();
            System.out.println("我是乡级人民政府,接收到国务院发送的消息:"+data);
            data = "发送2014年农田补贴,每亩地补贴168元";
            //修改广播传递的数据
            setResultData(data);
        }
    }
    

配置文件:

      <receiver android:name="com.itheima.gov.ProvinceBroadcastReceiver">
        <intent-filter android:priority="1000">
            <action android:name="com.itheima.broadcast.GWYFFNTBT"/>
        </intent-filter>
    </receiver>

09_服务和进程优先级

windows下的服务:没有界面,可以长期运行在后台的应用程序;
Android下的服务:没有界面,可以长期运行在后台的应用程序(没有界面的activity);

进程:是运行应用程序的载体。当应用程序一启动的时候,系统自动创建一个进程,负责运行dalvik虚拟机,我们开发的应用程序试运行在dalvik虚拟机上的。


Activity、BroadcastReceiver、Service、ContentProvider;

应用程序:四大组件放在一起就是一个应用程序;

解决方案:给创建的每个进程定义一个类型和优先级;

进程的生命周期: 1、当应用程序已启动的时候,进程被创建; 2、当手工结束进程,或者强制停止应用程序可以结束进程、系统内存空间不够使用,系统会自动杀死低优先级的进程;

进程的等级:

  1. Foreground process(前台进程)

    应用程序界面可见,用户正在操作,activity的onresume方法被执行了,可以相应点击事件。
    
  2. Visible process (可视进程)

    应用程序的ui界面,用户还可以看到,但是不能操作了。
    
  3. Service process (服务进程)

    应用程序没有界面,但是有一个后台的服务还处于运行状态
    
  4. Background process(后台进程)

    应用程序没有服务处于运行状态,应用程序被最小化了,activity执行了onstop方法
    
  5. Empty process (空进程)

    没有任何组件运行,所有的activity都关闭了,任务栈清空了。
    

10_服务的特点

特点:

1、服务第一次被开启的时候,系统创建一个服务对象,调用onCreate、onStartCommand;
2、服务职能被创建一次,可以被开启多次,多次开启只会调用onStartCommand;
3、停止服务调用onDestroy方法,服务只能被停止一次;
4、服务没有界面,可以长期运行在后台;

服务的应用场景:
    当需要在后台运行一段程序,程序需要与服务器端通讯;

11_电话窃听器的模板代码(重点)

在后台运行一段程序,监听电话的状态,当电话铃声响起的时候,创建一个录音机;当电话接通的时候,使用录音机开始录音;当电话挂断的时候,录音机停止录音,上传录制的音频文件;

实现步骤:

1、写一个服务运行在后台;
2、在服务里写一个电话的监听器;
3、当电话铃声响起的时候,准备一个录音机;
4、当电话接通的时候,使用录音机开始录音;
5、当电话挂断的时候,录音机停止录音,上传录制的音频文件;

代码:

package com.itheima.phonelistener;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.telephony.PhoneStateListener;
import android.telephony.TelephonyManager;
/**
 * 1、写一个服务运行在后台;
 */
public class PhoneStateListenerService extends Service {

@Override
public IBinder onBind(Intent intent) {
    // TODO Auto-generated method stub
    return null;
}

@Override
public void onCreate() {
    super.onCreate();

    //2、在服务里写一个电话的监听器;
    //得到系统提供的电话服务
    TelephonyManager tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);

    //监听电话的呼叫状态
    tm.listen(new MyListener(), PhoneStateListener.LISTEN_CALL_STATE);

}

private class MyListener extends PhoneStateListener{

    /**
     * 当电话状态发生变化时调用这个方法
     * state 电话状态:闲置、铃声响起、接通电话
     * incomingNumber 呼入的电话号码
     */
    @Override
    public void onCallStateChanged(int state, String incomingNumber) {
        super.onCallStateChanged(state, incomingNumber);

        switch (state) {
        case TelephonyManager.CALL_STATE_IDLE://闲置
            //5、当电话挂断的时候,录音机停止录音,上传录制的音频文件;
            break;

        case TelephonyManager.CALL_STATE_RINGING://铃声响起
             //3、当电话铃声响起的时候,准备一个录音机;

        break;

        case TelephonyManager.CALL_STATE_OFFHOOK://接通电话
            //4、当电话接通的时候,使用录音机开始录音;

            break;
        default:
            break;
        }
    }

}

}

12_利用服务实现电话窃听器

代码:

package com.itheima.phonelistener;

import java.io.IOException;

import android.app.Service;
import android.content.Intent;
import android.media.MediaRecorder;
import android.os.Environment;
import android.os.IBinder;
import android.telephony.PhoneStateListener;
import android.telephony.TelephonyManager;

/**
 * 1、写一个服务运行在后台;
 */
public class PhoneStateListenerService extends Service {

private MediaRecorder r;

@Override
public IBinder onBind(Intent intent) {
    // TODO Auto-generated method stub
    return null;
}

@Override
public void onCreate() {
    super.onCreate();

    //2、在服务里写一个电话的监听器;
    //得到系统提供的电话服务
    TelephonyManager tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);

    //监听电话的呼叫状态
    tm.listen(new MyListener(), PhoneStateListener.LISTEN_CALL_STATE);
}

private class MyListener extends PhoneStateListener{
    /**
     * 当电话状态发生变化时调用这个方法
     * state 电话状态:闲置、铃声响起、接通电话
     * incomingNumber 呼入的电话号码
     */
    @Override
    public void onCallStateChanged(int state, String incomingNumber) {
        super.onCallStateChanged(state, incomingNumber);

        switch (state) {
        case TelephonyManager.CALL_STATE_IDLE://闲置
             System.out.println("==========CALL_STATE_IDLE=========");
            //5、当电话挂断的时候,录音机停止录音,上传录制的音频文件;
             if(r != null){
                 r.stop();

                    r.release();
                    r = null;
             }
            break;

        case TelephonyManager.CALL_STATE_RINGING://铃声响起
             try {
                 System.out.println("==========CALL_STATE_RINGING=========");
                r = new MediaRecorder();
                r.setAudioSource(MediaRecorder.AudioSource.MIC);
                r.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
                r.setOutputFile(Environment.getExternalStorageDirectory()+"/info.3gp");
                r.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
                //准备录音机
                r.prepare();
            } catch (Exception e) {
                e.printStackTrace();
            }
        break;

        case TelephonyManager.CALL_STATE_OFFHOOK://接通电话
            //4、当电话接通的时候,使用录音机开始录音;
             System.out.println("==========CALL_STATE_OFFHOOK=========");
            r.start();
            break;
        }
    }
}
}

配置文件:

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<uses-permission android:name="android.permission.RECORD_AUDIO"/>

<service android:name="com.itheima.phonelistener.PhoneStateListenerService"></service>
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值