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) 方法可保证服务在设备重启后也能按计划运行。