安卓权威编程指南-笔记(第26章 服务的作用)

Posted mystic

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了安卓权威编程指南-笔记(第26章 服务的作用)相关的知识,希望对你有一定的参考价值。

1. IntentService

IntentService也是一个context(Service是Context的子类),并能够响应intent。

一个最基本的IntentService实例如下:

public class PollService extends IntentService {
    private static final String TAG = "PollService";
    public static Intent newIntent(Context context) {
    return new Intent(context, PollService.class);
}
public PollService() {
    super(TAG);
}
@Override
protected void onHandleIntent(Intent intent) {
    Log.i(TAG, "Received an intent: " + intent);
    }
}

服务的intent又称作命令,每一个命令都要求服务完成某项具体的任务,服务种类不同,其执行命令的方式也不尽相同。

接收到首个命令时, IntentService 完成启动,并触发一个后台线程,然后将命令放入队列。随后, IntentService 继续按顺序执行每一条命令,并针对每一条命令在后台线程上调用onHandleIntent(Intent) 方法。新进命令总是放置在队列尾部。最后,执行完队列中的全部命令后,服务也随即停止并被销毁。

2.检查网络的可用性

  

private boolean isNetworkAvailableAndConnected() {
    ConnectivityManager cm =(ConnectivityManager) getSystemService(CONNECTIVITY_SERVICE);
    boolean isNetworkAvailable = cm.getActiveNetworkInfo() != null;
    boolean isNetworkConnected = isNetworkAvailable &&
    cm.getActiveNetworkInfo().isConnected();
    return isNetworkConnected;
}
//获取网络状态需要增加权限
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

 

 

3.Notification

 1  protected void onCreate(Bundle savedInstanceState) {
 2         super.onCreate(savedInstanceState);
 3         setContentView(R.layout.activity_main);
 4 
 5         final Button send_notice = (Button)findViewById(R.id.send_notice);
 6         send_notice.setOnClickListener(new View.OnClickListener() {
 7             @Override
 8             public void onClick(View v) {
 9                 //1.获取NotificationManager对象
10                 NotificationManager manager = (NotificationManager)getSystemService(NOTIFICATION_SERVICE);
11                 //设置意图
12                 Intent intent = new Intent(MainActivity.this,NotificationActivity.class);
13                 //通过PengdingIntent.getActivity得到PendingIntent对象
14                 PendingIntent pendingIntent = PendingIntent.getActivity(MainActivity.this,0,intent,0);
15                 Notification notification = new Notification.Builder(MainActivity.this)
16                         .setContentTitle("1")
17                         .setContentText("?")
18                         .setContentIntent(pendingIntent) //设置ContentIntent 传入PendingIntent 设置点击事件
19                         .setWhen(System.currentTimeMillis())
20                         .setSmallIcon(R.mipmap.ic_launcher)
21                         .setLargeIcon(BitmapFactory.decodeResource(getResources(),R.mipmap.ic_launcher))
22 
23                         .setAutoCancel(true)//当点击了这个通知的时候 通知会自动取消掉 取消通知的第二种方法 调用得到的Manager对象的cancle方法 cancle方法中传入创建通知时给定的Id
24 
25                         .setSound(Uri.fromFile(new File("/system/media/audio/ringtones/Luna.ogg")))
26 
27                         //设置手机静止和震动的时长,以毫秒为单位,以0开始 奇数为震动的时长 偶数为静止的时长 需要声明权限
28                         .setVibrate(new long[]{0,1000,1000,1000})
29 
30                         /*
31                         *控制手机Led灯
32                         * 第一个参数用于指定Led灯的颜色 第二个参数用于指定LED灯亮起的时长,以毫秒为单位,第三个参数用于指定LED灯暗去的时长
33                          */
34                         .setLights(Color.RED,1000,1000)
35 
36                         //使用通知的默认效果 会根据当前手机环境来决定播放什么铃声以及如何震动
37 //                        .setDefaults(NotificationCompat.DEFAULT_ALL)
38 
39                         /*
40                         *通过setStyle方法在通知中显示一段长文字
41                         *创建一个NotificationCompat.BigTextStyle对象,调用它的bigText方法将文字传入即可
42                          */
43                         .setStyle(new Notification.BigTextStyle().bigText("Learn NotificaionLearn NotificaionLearn NotificaionLearn Notific" +
44                                 "aionLearn NotificaionLearn NotificaionLearn NotificaionLearn NotificaionLearn NotificaionLear" +
45                                 "n NotificaionLearn Notificaion Learn Notificaion"))
46 
47                         //显示一张大图片
48                         .setStyle(new Notification.BigPictureStyle().bigPicture(BitmapFactory.decodeResource(getResources(),R.mipmap.ic_launcher)))
49 
50                         //设置优先级
51                         .setPriority(NotificationCompat.PRIORITY_MAX)
52 
53 
54                         .build();
55                 //调用notify 方法将Notification显示出来
56                 manager.notify(1,notification);
57 
58 
59             }
60         });

 

 

4.服务

4.1 服务的能与不能

  与activity一样,服务是一个有生命周期回调方法的应用组件,这些回调方法同样也会在主线程上运行。

  初始创建的服务不会在后台线程上运行任何代码,这也是我们推荐使用IntentService的最主要原因,大多数重要服务都需要某种后台进程,而IntentService已提供了一套标准实现代码。

 

4.2 服务的生命周期

  如果是startService(Intent)方法启动的服务,其生命周期很简单,并具有三种周期回调方法。  

  •   onCreate方法:服务创建时调用。
  •   onStartCommand(Intent, int, int )方法:每次组件通过startService(intent)方法启动时调用一次。它有两个整数参数,一个是标识符集,一个是启动ID。标识符集用来表示当前intent发送究竟是一次重新发送,还是一次从没成功的发送,每次调用onStartCommand(Intent , int ,int )方法,启动ID都会不同,因此,启动ID也可用于区分不同的命令。
  •   onDestroy()方法:服务不再需要时调用,通常是在服务停止后。

  服务停止时会调用onDestroy()方法,服务停止的方式取决于服务的类型。服务的类型由onStartCommand()方法的返回值确定,可能的服务类型有Service.START_NOT_STICKY、START_READLIVER_INTENT和START_STICKY.

 

4.3 non-sticky 服务

  IntentService是一种non-sticky服务。non-sticky服务在服务自己认为自己已完成任务时停止。为获得non-sticky服务,应返回START_NOT_STICKY或START_REDELIVER_INTENT.

  通过调用stopSelf()或stopSelf(int)方法,我们告诉Android任务已完成。stopself()是个无条件方法。不管onStartCommand()方法调用多少次。该方法总会成功停止服务。

  stopSelf(int) 是个有条件的方法。该方法需要来自于 onStartCommand(...) 方法的启动ID。只有在接收到最新启动ID后,该方法才会停止服务。(这也是 IntentService 的后台工作原理。)

    返回 START_NOT_STICKY 和 START_REDELIVER_INTENT 有什么不同呢?区别就在于,如果系统需要在服务完成任务之前关闭它,则服务的具体表现会有所不同。 START_NOT_STICKY 型服务说消亡就消亡了;而 START_REDELIVER_INTENT 型服务则会在资源不再吃紧时,尝试再次启动服务。    

    选择 START_NOT_STICKY 还是 START_REDELIVER_INTENT ,这要看服务对应用有多重要了。如果不重要,就选择 START_NOT_STICKY 。

 

4.4 sticky 服务

  sticky服务会持续运行,直到外部组件调用Context.stopService(Intent)方法让它停止。为获得sticky服务,应返回START_STICKY。

    Sticky服务启动后会持续运行,除非某个组件调用Context.stopService(Intent)方法停止它。

    可传入一个null intent 给 onStartCommand()方法,实现服务的重启。

    sticky服务适用与长时间运行的服务。

 

4.5 绑定服务

除以上各类服务外,也可使用 bindService(Intent,ServiceConnection, int) 方法绑定一个服务,以此获得直接调用绑定服务方法的能力。 ServiceConnection 是代表服务绑定的一个对象。它负责接收全部绑定回调方法。

fragment中,绑定代码示例代码如下:

private ServiceConnection mServiceConnection = new ServiceConnection() {
     public void onServiceConnected(ComponentName className,IBinder service) {
    // Used to communicate with the service
    MyBinder binder = (MyBinder)service;
    }
    public void onServiceDisconnected(ComponentName className) {
    }
 };
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Intent i = new Intent(getActivity(), MyService.class);
        getActivity().bindService(i, mServiceConnection, 0);
    }
    @Override
    public void onDestroy() {
        super.onDestroy();
        getActivity().unbindService(mServiceConnection);
    }
}

对服务来说,绑定引入了另外两个生命周期回调方法。
? onBind(Intent) 方法:每次绑定服务时调用,返回来自 ServiceConnection.onService
Connected(ComponentName,IBinder) 方法的 IBinder 对象。
? onUnbind(Intent) 方法:服务绑定终止时调用。

 

 

4.5.1. 本地服务绑定
MyBinder 是怎样一种对象呢?如果服务是个本地服务, MyBinde 很可能就是本地进程中的一
个简单Java对象。通常, MyBinde 用于提供一个句柄,以便直接调用服务方法:

private class MyBinder extends IBinder {
    public MyService getService() {
    return MyService.this;
    }
}
@Override
public void onBind(Intent intent) {
    return new MyBinder();
}

这种模式看上去让人激动。这是Android系统中唯一一处支持组件间直接对话的地方。不过,
我们并不推荐此种模式。服务是种高效的单例,与仅使用一个单例相比,使用此种模式显现不出
优势。

4.5.2. 远程服务绑定
绑定更适用于远程服务,因为它们赋予了其他进程中应用调用服务方法的能力。

 

5. JobScheduler和JobService

使用AlarmManager、IntentService和PendingIntent相互配合,创走周期性的后台任务,实现一个完全可用的后台服务还需要手动执行以下操作。

?  计划一个周期性任务

?  检查周期性任务的运行状态

?  检查网络是否可用

在实际场景下,还有更多想法需要实现,例如请求失败,是否还需要稍后重试机制。或者是只允许应用使用不限量的网络连接...

在系统控制方面,本章实现的后台服务也存在一些问题,它无法在某种情况停下来,除非手动停止。

 

 

JobScheduler:除了实现常规后台任务之外,JobScheduler还支持按场景、按条件运行后台服务。

 

JobScheduler的使用:

创建一个类继承JobService,并覆盖onStartJob(JobParameters parms)方法和onStopJob(JobParameters params)。

public class PollService extends JobService {


    @Override
    public boolean onStartJob(JobParameters params) {
        return false;
    }

    @Override
    public boolean onStopJob(JobParameters params) {
        return false;
    }
}

Android准备好执行任务时,服务就会启动,此时会在主线程上收到onStartJob()方法调用。该方法返回false结果表示:"交代的任务我已全力去做,现在做完了。”返回 true 结果则表示:“任务收到,正在做,但是还没有做完。”

JobService在执行的时候需要单开线程,可以使用AsyncTask按如下方式创建新线程。

    private PollTask mCurrentTask;

@Override
    public boolean onStartJob(JobParameters parms){ 
        mCurrentTask = new PollTask();
        mCurrentTask.execute(parms);
        return true;
    }    


private class PollTask extends AsyncTask<JobParameters,Void,Void> {
    @Override
    protected Void doInBackground(JobParameters... params) {
        JobParameters jobParams = params[0];
        //执行任务的逻辑
        jobFinished(jobParams, false);  
        return null;
    }
}    

任务执行完毕后,就可以调用jobFinished(JobParameters, boolean)方法通知结果,如果该方法的第二个参数传入true的话,就等于说:“事情这次做不完了,请计划在下次某个时间继续吧。”

 

onStopJob(JobParameters)方法适合在中断任务时调用,用户通常需要服务在有WIFI连接时才运行,如果在调用JobFinished()之前(任务完成之前),手机就没了Wifi,onStopJob(...) 方法就会被调用,也就是说,一切任务就立即停止了。

@Override
public boolean onStopJob(JobParameters params) {
    if (mCurrentTask != null) {
    mCurrentTask.cancel(true);
    }
    return true;
}

调用 onStopJob(...) 方法就是表明,服务马上就要停掉了。不要抱有幻想,请立即停止手头上的一切事情。这里,返回 true 表示:“任务应该计划在下次继续。”返回 false 表示:“不管怎样,事情就到此结束吧,不要计划下次了。”

 

使用JobService,必须在Manifest配置清单中添加权限:

<service
android:name=".PollService"
android:permission="android.permission.BIND_JOB_SERVICE"  //添加的权限控制只有JobScheduler才能运行它。
android:exported="true"/>

 

通过JobScheduler检查是否已计划好了任务:

final int JOB_ID = 1;
JobScheduler scheduler = (JobScheduler)
    context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
boolean hasBeenScheduled = false;
for (JobInfo jobInfo : scheduler.getAllPendingJobs()) {
    if (jobInfo.getId() == JOB_ID) {
    hasBeenScheduled = true;
    }
}

 

PoolService的运行:

final int JOB_ID = 1;

JobScheduler scheduler = (JobScheduler)
    context.getSystemService(Context.JOB_SCHEDULER_SERVICE);

    JobInfo jobInfo = new JobInfo.Builder(JOB_ID, new ComponentName(context, PollService.class))
    .setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED)
    .setPeriodic(1000 * 60 * 15)
    .setPersisted(true)
    .build();
scheduler.schedule(jobInfo);

上述代码计划任务每15分钟运行一次,但前提条件是有Wi-Fi或有可用的不限流量网络。调用 setPersisted(true) 方法可保证服务在设备重启后也能按计划运行。

 













以上是关于安卓权威编程指南-笔记(第26章 服务的作用)的主要内容,如果未能解决你的问题,请参考以下文章

安卓权威编程指南-笔记(第25章 搜索)

安卓权威编程指南-笔记(第21章 XML drawable)

安卓权威编程指南-笔记(第27章 broadcast intent)

安卓权威编程指南-笔记(第24章 Looper Handler 和 HandlerThread)

安卓权威编程指南-笔记 (第29章定制视图与触摸事件)

安卓权威编程指南 - 第五章学习笔记(两个Activity)