Android开发高级进阶——Service与Notification

众所周知,Service是Android中实现程序后台运行的解决方案,非常适合于执行不需要和用户交互而且要求长期运行的任务。Notification是显示在手机状态栏的通知,通过Notification和startForeground可以将Service后台服务设置为前台服务。

一. Service的分类


按种类分为LocalService和RemoteService

  • LocalService:当前进程的Service,依附在主进程上,节约资源,通信不需要进程间通信,但主进程被杀掉时LocalService也会被停止。一般可以做播放操作。
  • RemoteService:远程进程的Service,是一个独立的进程,当主进程被杀掉时,其服务依然会运行,比较灵活,但使用复杂。可以做一些系统常驻进程或守护进程(当主进程被杀掉时,可以用守护进程唤醒主进程)。

按类型分为前台Service和后台Service

  • 前台Service:在通知栏一直显示的服务,最大程度保证服务不被杀掉。
  • 后台Service:默认的服务为后台服务,看不见。

前台Service的优先级高于后台Service。

启动方式有startService和bindService

  • startService:onCreate只会在Service第一次创建的时候被调用,start一次之后,下次不会再onCreate了,每次会执行onStartCommand。
  • bindService:Service中的onBind是与外界进行关联的。onBind返回的IBinder都会在与其他组件连接时返回,即IBinder是一个负责沟通的桥梁。

二. Service、Thread、Process


Service是处理一些后台任务,在主线程中,并不能执行耗时操作。而Thread是开启一个子线程,在子线程中执行耗时操作,这样不会阻塞主线程。

那么Service与Thread、Process有什么关系呢?

其实它们之间并无太大关联,只是Service可以放在其他进程中,Service是进程中的,Thread也是进程中的,Service中耗时操作又可以在通过Thread执行。Service中的IBinder可以帮助我们与其他进程进行通信。

获取当前进程:Thread.currentThread().getId();

在Service中创建一个子线程与Activity中创建一个子线程有什么区别?

当Activity被销毁,就无法获取在被销毁的Activity中创建的子线程实例了。例如,Activity创建了一个子线程在后台运行,执行完之后,返回数据时,创建它的Activity已被销毁,这是线程不安全的。再例如,Activity1创建了一个子线程,Activity2是无法对Activity1创建的子线程进行操作。 而Service是一个服务,所有的Activity都可以与该Service进行关联,即使Activity被销毁,但依然可以获取Service中的IBinder的实例。 所以,用Service处理后台任务,Activity就可以放心的被finish掉了,完全不用担心对后台任务无法进行控制。

三. AndroidManifest.xml中Service的一些标签


  • name:服务的类名。
  • label:服务名,如果不设置,默认为类名。
  • icon:服务的图标。
  • permission:服务的权限。
  • process:服务的进程名。
  • enabled:boolean类型,true表示该服务默认被系统启动。
  • exported:表示该服务是否能够被其他应用程序控制或连接,是否向外通过服务。

四. Notification


  1. 创建通知 可以在NotificationCompat.Builder对象中为通知指定 UI 信息和操作。要创建通知,请调用NotificationCompat.Builder.build(),它将返回包含具体规范的Notification对象。要发出通知,请通过调用NotificationManager.notify()将 [Notification对象传递给系统。
  2. 必需的通知内容 Notification对象必须包含以下内容:
  • 小图标,由setSmallIcon()设置。
  • 标题,由setContentTitle()设置。
  • 详细文本,由setContentText()设置。
  1. 通知操作 尽管通知操作都是可选的,但是至少应向通知添加一个操作。 操作允许用户直接从通知转到应用中的Activity,用户可在其中查看一个或多个事件或执行进一步的操作。 一个通知可以提供多个操作。应该始终定义一个当用户点击通知时会触发的操作;通常,此操作会在应用中打开Activity。也可以向通知添加按钮来执行其他操作,例如,暂停闹铃或立即答复短信。 在Notification内部,操作本身由PendingIntent定义,后者包含在应用中启动Activity的Intent。要将PendingIntent与手势相关联,请调用NotificationCompat.Builde的适当方法。例如,如果要在用户点击抽屉式通知栏中的通知文本时启动Activity,则可通过调用setContentIntent()来添加PendingIntent。

创建简单通知

以下代码段说明了一个指定某项 Activity 在用户点击通知时打开的简单通知。 请注意,该代码将创建TaskStackBuilder对象并使用它来为操作创建PendingIntent。

NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(this)
        .setSmallIcon(R.drawable.notification_icon)
        .setContentTitle("My notification")
        .setContentText("Hello World!");
// Creates an explicit intent for an Activity in your app
Intent resultIntent = new Intent(this, ResultActivity.class);

// The stack builder object will contain an artificial back stack for the
// started Activity.
// This ensures that navigating backward from the Activity leads out of
// your application to the Home screen.
TaskStackBuilder stackBuilder = TaskStackBuilder.create(this);
// Adds the back stack for the Intent (but not the Intent itself)
stackBuilder.addParentStack(ResultActivity.class);
// Adds the Intent that starts the Activity to the top of the stack
stackBuilder.addNextIntent(resultIntent);
PendingIntent resultPendingIntent =
        stackBuilder.getPendingIntent(0, PendingIntent.FLAG_UPDATE_CURRENT);
mBuilder.setContentIntent(resultPendingIntent);
NotificationManager mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
// mId allows you to update the notification later on.
mNotificationManager.notify(mId, mBuilder.build());

五. 补充

  1. 旋转屏幕时,Activity会被重新创建,bindService的连接会被断开。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏向治洪

Android 进程回收之LowMemoryKiller原理篇

在前面的文章Android进程保活一文中,对于LowMemoryKiller的概念做了简单的提及。LowMemoryKiller简称低内存杀死机制。简单来说,L...

4827
来自专栏猿天地

Spring Cloud中如何保证各个微服务之间调用的安全性

一.背景 微服务架构下,我们的系统根据业务被拆分成了多个职责单一的微服务。 每个服务都有自己的一套API提供给别的服务调用,那么如何保证安全性呢? 不是说你想调...

59014
来自专栏24K纯开源

Ubuntu13.04配置:Vim+Syntastic+Vundle+YouCompleteMe

序言     使用Ubuntu和vim已经有一段时间了,对于Vim下的插件应用,我总是抱着一股狂热的态度。这次,又在网上闲逛着,发现了一个个人博客提到了Vim代...

26710
来自专栏CodingBlock

Android查缺补漏--BroadcastReceiver的类型与使用

Broadcast 是一种被用于应用内和应用之间传递信息的机制。一个广播可以对应多个接受者。一个完整的广播机制,需要具有以下三个要素: 发送广播的Broadca...

1957
来自专栏www.96php.cn

thinkphp实现模板切换

thinkphp实现模板切换 首先在后台某个接口读取模板 Admin\Common\function.php /*  * 获取模板目录  */ function...

3185
来自专栏安恒网络空间安全讲武堂

打个3dsctf,电脑机都开不了?

3DSCTF Malware 3道题 writeup tags: - Reverse - malware ---- 这个ctf有个Malware的题型,...

2489
来自专栏网络

代理工具shadowProxy之神出鬼没的切换IP地址

前言 在渗透测试过程中,往往会遇到特别“小气”的目标,稍微碰一下就封IP。这种情况下,我们很自然的想到通过网上大量的免费代理进行IP隐匿。 那么问题来了,难道拿...

3038
来自专栏数据和云

New in 12.2- 让每个PDB都拥有自己的 local undo表空间

12.1引入temp undo概念 ,12.2引入local undo,也就是每个pdb拥有自己的undo表空间(以前版本共用一个undo表空间),rac中每个...

3655
来自专栏于晓飞的专栏

Ubuntu开发环境配置

每次电脑切换都需要重新装一次系统,重新装系统后还需要安装各种各样的工具。如果每次都到网络上搜索工具的安装方法,就要浪费一大笔时间。这里把我常用的工具的安装方法分...

3622
来自专栏wOw的Android小站

[Android] 针对生成的图片文件在系统Gallery不显示的处理

  之前遇到过一个问题,就是发现我在程序中生成一个新的 Bitmap 之后,当我打开系统的 Gallery 查看时,并没有看到新生成的图像。然而打开文件浏览器,...

891

扫码关注云+社区