详解Android中的Service

Service简介:

Service是被设计用来在后台执行一些需要长时间运行的操作。
Android由于允许Service在后台运行,甚至在结束Activity后,因此相对来说,Service相比Activity拥有更高的优先级。

创建Service:

要创建一个最基本的Service,需要完成以下工作:1)创建一个Java类,并让其继承Service 2)重写onCreate()和onBind()方法

其中,onCreate()方法是当该Service被创建时执行的方法,onBind()是该Service被绑定时执行的方法。

public class ExampleService extends Service{
  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
  @Override
  public void onCreate() {
    super.onCreate();
  }
}

当创建了一个新的Service后,还必须在AndroidManifest.xml文件中对他进行配置,需要在application节点内包含一个Service标记。

<service android:name=".ExampleService" android:enabled="true" android:permission="exam02.chenqian.com.servicedemo"></service>

当然,如果你想要你自定义的Service仅能被自己编写的该应用程序使用,还可以在标签内添加:

android:permission="exam02.chenqian.com.servicedemo"

让Service执行特定的任务:

如果想要Service执行特定的任务,可以复写Service的onStartCommand()方法,注意在API15之前为onStart()方法,现已不推荐,onStartCommand()方法的执行为该Service onCreate()之后。

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
  return super.onStartCommand(intent, flags, startId);
}

启动和停止Service:

显式启动一个Service:

// 显示启动ExampleService
Intent intent = new Intent(this,ExampleService.class);
// 启动ExampleService
startService(intent);

为了方便观察,我们可以在之前创建的自定义的Service类中的onStartCommand()方法中添加Log.i("ServiceState","-------------->is Running");

当我们从MainActivity调用运行时,可以在Logcat中观察到输出: I/ServiceState: is Running
当然,我们也可以停止一个Service,为了让我们更清晰的观察到效果,我们可以在ExampleService类中复写onDestroy()方法:

  @Override
  public void onDestroy() {
    Log.i("ServiceState","------------------>Destroy");
    super.onDestroy();
  }

可以在MainActivity中通过以下方式停止一个Service:

显示停止一个Service:注意,写这里时更换了一个Service,并将该自定义的Service定位MyService,已经不是之前的ExampleService,不过您认可按照自己之前的继续编写,毕竟方法都是一样的;-)

//显示关闭Service
Intent serviceIntent = new Intent(MainActivity.this,MyService.class);
//关闭Service
stopService(serviceIntent);

注意Service的调用不可嵌套,因此无论Service被调用了多少次,对stopService()停止的一次调用就会终止它所匹配运行中的Service。

由于Service具有较高的优先级,通常不会被运行时终止,因此可以通过自终止来避免后台运行Service耗费系统的资源。具体方法为在onStartCommand()方法中加入stopSelf();但是要注意的是这里的stopSelf()并不是直接终止Service,而是当Service的所有功能或请求执行完后,将Service停止掉,而不是等待系统回收,停止会调用onDestroy()销毁该Service。

将Service绑定到Activity:

当一个Service在一个Activity中被调用的时候,并不会随着Activity的销毁而销毁,而是仍有可能继续在后台运行着继续占用系统的资源,因此如果实现当Activity销毁时自动停止与其相关的服务,将会极大的节约系统的资源占用,我们可以通过以下方式实现Activity与Service的绑定:

XML布局文件:在该布局文件中实现了四个按钮,分别执行启动Service、停止Service、绑定Service和解除绑定Service,清楚了吧:-)

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:id="@+id/activity_main"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical"
  tools:context="demo.chenqian.com.androidserverdemo.MainActivity">
  <!-- 开启Service -->
  <Button
    android:id="@+id/btnStartService"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="20dp"
    android:text="@string/startService"/>
  <!-- 关闭Service -->
  <Button
    android:id="@+id/btnStopService"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="20dp"
    android:text="@string/stopService"/>
  <!-- 绑定Service -->
  <Button
    android:id="@+id/btnBindService"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="20dp"
    android:text="@string/bindService"/>
  <!-- 解绑Service -->
  <Button
    android:id="@+id/btnUnbindService"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="20dp"
    android:text="@string/unbindService"/>
</LinearLayout>

MyService类:

package demo.chenqian.com.androidserverdemo;
import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.support.annotation.Nullable;
import android.util.Log;
public class MyService extends Service{
  /* 1、在下方我们定义了内部类MyBinder,这就是为什么我们这里现在能定义binder的原因
    2、这里我们定义binder成员变量的目的是为了在下文的MainActivity中实现转型*/
  private MyBinder binder = new MyBinder();
  @Override
  public void onCreate() {
    Log.d("ServiceInfo","创建成功");
    super.onCreate();
  }
  @Nullable
  @Override
  public IBinder onBind(Intent intent) {
    Log.d("ServiceInfo","绑定成功");
    return null;
  }
  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
    Log.d("ServiceInfo","开始执行");
    return super.onStartCommand(intent, flags, startId);
  }
  @Override
  public boolean onUnbind(Intent intent) {
    Log.d("ServiceInfo","解绑成功");
    return super.onUnbind(intent);
  }
  @Override
  public void onDestroy() {
    Log.d("ServiceInfo","销毁成功");
    super.onDestroy();
  }

  /*我们知道Android系统为了安全防护和整体的稳定性,将每一个应用程序隔离在相应的独立的“沙盒”之中,因此我们自定义的Service实际上是运行在

    用户空间的,那么我们又有许多服务需要引用系统的Service,那么一个在用户空间一个系统空间,他们之间如何实现通信呢,这就需要Binder了,

   Binder是Android系统中实现不同进程之间通信的一种方式,Binder本身有粘合剂的意思,Binder可以粘合Android系统中的四大组件,因此下方我 们在MyService类中新建了一个MyBinder内部类,并让其继承Binder类,用来实现对MyService的获取,这样,你该知道为什我们上文要新建一个My-Binder的成员变量了吧 ^_^,在下方的MainActivity中你也可以看到相关实例的运用,

例如

public void
onServiceConnected(ComponentName
   name, IBinder service),注意这里的service是IBinder类型的,我们下方获取MyService将会用到他*/
  class MyBinder extends Binder{
    MyService getService(){
      Log.d("ServiceInfo","成功得到当前服务实例");
      return MyService.this;
    }
  }
}

MainActivity类:

package demo.chenqian.com.androidserverdemo;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Binder;
import android.os.IBinder;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
  private Context mContext;
  private Button btnStartService;
  private Button btnStopService;
  private Button btnBindService;
  private Button btnUnbindService;
  private MyService myService;
  private Intent serviceIntent;
  private boolean isBond;
  /*isBond该变量用来标识当前的Activity与Service是否正在绑定,因为如果不进行标识,如果Activity没有
   与Service进行绑定,而执行解除绑定的操作,会照成错误或抛出异常,因为当接触绑定时Android不允许绑定为null */
  /*注意,这里我们新建了一个connection并重写了相关方法,为什么我们要新建这个连接,那是因为在下方的绑定和解绑
   方法即bind和unbind需要一个connection。我们复写相关方法一是为了方便观察,另一个是为了在连接成功或失去关闭
   连接时,执行相关的自定义的任务或操作*/
  private ServiceConnection connection = new ServiceConnection() {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
      Log.d("ServiceState","连接成功");
      myService = ((MyService.MyBinder)service).getService();
    }
    @Override
    public void onServiceDisconnected(ComponentName name) {
      Log.d("ServiceState","关闭连接");
       //当连接指向实例为null没有指引的连接的实例时,好被虚拟机回收,降低占用的资源
      myService = null;
    }
  };
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    //初始化数据
    mContext = this;
    isBond = false;
    //引入需要用到的组件
    btnStartService = (Button) findViewById(R.id.btnStartService);
    btnStopService = (Button) findViewById(R.id.btnStopService);
    btnBindService = (Button) findViewById(R.id.btnBindService);
    btnUnbindService = (Button) findViewById(R.id.btnUnbindService);
    //为按钮添加单击事件
    btnStartService.setOnClickListener(this);
    btnStopService.setOnClickListener(this);
    btnBindService.setOnClickListener(this);
    btnUnbindService.setOnClickListener(this);
  }
  @Override
  protected void onStart() {
    serviceIntent = new Intent(this,MyService.class);
    super.onStart();
  }
  @Override
  public void onClick(View v) {
    switch (v.getId()){
      case R.id.btnStartService:
        //开启Service
        startService(serviceIntent);
        break;
      case R.id.btnStopService:
        //关闭Service
        stopService(serviceIntent);
        break;
      case R.id.btnBindService:
        //绑定Service
        isBond = bindService(serviceIntent,connection,Context.BIND_AUTO_CREATE);
        break;
      case R.id.btnUnbindService:
        //解绑Service,当连接为null是解绑会报错
        if(isBond){
          unbindService(connection);
          //如果解绑成功,则修改连接标识为假
          isBond = false;
        }
        break;
    }
  }
}

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="demo.chenqian.com.androidserverdemo">
  <application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <activity android:name=".MainActivity">
      <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
      </intent-filter>
    </activity>
    <service android:name=".MyService" android:enabled="true" android:permission="demo.chenqian.com.androidserverdemo"></service>
  </application>
</manifest>

 关于以后:

1、感觉Binder那块还给大家解释的不太清楚,以后再深入研究下补充完整

2、有时间会编写一个简单的后台播放音乐的实例提供给大家参考一下

以上所述是小编给大家介绍的详解Android中的Service,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

(0)

相关推荐

  • Android基于service实现音乐的后台播放功能示例

    本文实例讲述了Android基于service实现音乐的后台播放功能.分享给大家供大家参考,具体如下: Service是一个生命周期长且没有用户界面的程序,当程序在各个activity中切换的时候,我们可以利用service来实现背景音乐的播放,即使当程序退出到后台的时候,音乐依然在播放.下面我们给出具体例子的实现: 当然,首先要在资源文件夹中添加一首MP3歌曲: 要实现音乐的播放,需要在界面中放置两个按钮,用来控制音乐的播放和停止,通过使用startService和stopService来实现

  • Android Service中使用Toast无法正常显示问题的解决方法

    本文实例讲述了Android Service中使用Toast无法正常显示问题的解决方法.分享给大家供大家参考,具体如下: 在做Service简单练习时,在Service中的OnCreate.OnStart.OnDestroy三个方法中都像在Activity中同样的方法调用了Toast.makeText,并在Acitivy中通过两个按钮来调用该服务的onStart和onDestroy方法: DemoService代码如下: @Override public void onCreate() { su

  • Android 如何保证service在后台不被kill

    Android 其实无法做多绝对的不被后台kill掉,我们只能尽量使用一些操作提升不被kill的机会. 一.onStartCommand方法,返回START_STICKY START_STICKY 在运行onStartCommand后service进程被kill后,那将保留在开始状态,但是不保留那些传入的intent.不久后service就会再次尝试重新创建,因为保留在开始状态,在创建     service后将保证调用onstartCommand.如果没有传递任何开始命令给service,那将

  • Android IntentService详解及使用实例

    Android IntentService详解 一.IntentService简介 IntentService是Service的子类,比普通的Service增加了额外的功能.先看Service本身存在两个问题: Service不会专门启动一条单独的进程,Service与它所在应用位于同一个进程中: Service也不是专门一条新线程,因此不应该在Service中直接处理耗时的任务: 二.IntentService特征 会创建独立的worker线程来处理所有的Intent请求: 会创建独立的wor

  • Android AccessibilityService实现微信抢红包插件

    在你的手机更多设置或者高级设置中,我们会发现有个无障碍的功能,很多人不知道这个功能具体是干嘛的,其实这个功能是为了增强用户界面以帮助残障人士,或者可能暂时无法与设备充分交互的人们 它的具体实现是通过AccessibilityService服务运行在后台中,通过AccessibilityEvent接收指定事件的回调.这样的事件表示用户在界面中的一些状态转换,例如:焦点改变了,一个按钮被点击,等等.这样的服务可以选择请求活动窗口的内容的能力.简单的说AccessibilityService就是一个后

  • android使用NotificationListenerService监听通知栏消息

    NotificationListenerService是通过系统调起的服务,在应用发起通知时,系统会将通知的应用,动作和信息回调给NotificationListenerService.但使用之前需要引导用户进行授权.使用NotificationListenerService一般需要下面三个步骤. 注册服务 首先需要在AndroidManifest.xml对service进行注册. <service android:name=".NotificationCollectorService&q

  • Android Service类与生命周期详细介绍

    Android  Service类与生命周期 Service是Android四大组件与Activity最相似的组件,都代表可执行的程序,区别在于Service一直在后台运行且没有用户界面. 1.Service的类图和生命周期 先来看看Service的类图: 接下来看看Service的生命周期: 2.开发Service (1)开发Service需要两步: 第1步:定义子类,继承Service 第2步:在AndroidManifest.xml文件中配置Service (2)创建Service pub

  • Android实现微信自动向附近的人打招呼(AccessibilityService)

    学习功能强大的AccessibilityService!!! 以下是本人根据自动抢红包的实现思路敲的用于微信自动向附近的人打招呼的核心代码 public class AutoService extends AccessibilityService implements View.OnClickListener { private static final String TAG = "test"; /** * 微信的包名 */ static final String WECHAT_PAC

  • Android Service的启动过程分析

    Android Service的启动过程分析 刚开始学习Service的时候以为它是一个线程的封装,也可以执行耗时操作.其实不然,Service是运行在主线程的.直接执行耗时操作是会阻塞主线程的.长时间就直接ANR了. 我们知道Service可以执行一些后台任务,是后台任务不是耗时的任务,后台和耗时是有区别的喔. 这样就很容易想到音乐播放器,天气预报这些应用是要用到Service的.当然如果要在Service中执行耗时操作的话,开个线程就可以了. 关于Service的运行状态有两种,启动状态和绑

  • 详解Android中的Service

    Service简介: Service是被设计用来在后台执行一些需要长时间运行的操作. Android由于允许Service在后台运行,甚至在结束Activity后,因此相对来说,Service相比Activity拥有更高的优先级. 创建Service: 要创建一个最基本的Service,需要完成以下工作:1)创建一个Java类,并让其继承Service 2)重写onCreate()和onBind()方法 其中,onCreate()方法是当该Service被创建时执行的方法,onBind()是该S

  • 详解Android中Service AIDL的使用

    目录 前言 Service基本用法--本地服务 远程服务 -- AIDL 服务端 客户端 前言 有些朋友可能是从事开发工作的时间不是特别的长,所以觉得Service相对与另外两个组件activity.broadcast receiver来说,使用可能并不是特别的多,所以对Service来说,理解不是特别的深入,只是有一个大概的概念,今天就和一块来走一下Service,希望能够帮助到大家对Service有更深入的理解. Service基本用法--本地服务 我们知道服务分为本地服务和远程服务,而本地

  • 详解Android中的ActivityThread和APP启动过程

    ActiviryThread ActivityThread的初始化 ActivityThread即Android的主线程,也就是UI线程,ActivityThread的main方法是一个APP的真正入口,MainLooper在它的main方法中被创建. //ActivityThread的main方法 public static void main(String[] args) { ... Looper.prepareMainLooper(); ActivityThread thread = ne

  • 详解Android中Intent对象与Intent Filter过滤匹配过程

    如果对Intent不是特别了解,可以参见博文<详解Android中Intent的使用方法>,该文对本文要使用的action.category以及data都进行了详细介绍.如果想了解在开发中常见Intent的使用,可以参见<Android中Intent习惯用法>. 本文内容有点长,希望大家可以耐心读完. 本文在描述组件在manifest中注册的Intent Filter过滤器时,统一用intent-filter表示. 一.概述 我们知道,Intent是分两种的:显式Intent和隐式

  • 详解 Android中Libgdx使用ShapeRenderer自定义Actor解决无法接收到Touch事件的问题

    详解 Android中Libgdx使用ShapeRenderer自定义Actor解决无法接收到Touch事件的问题 今天在项目中实现了一个效果,主要是画一个圆.为了后续使用方便,将这个圆封装在一个自定义Actor(CircleActot)中,后续想显示一个圆的时候,只要创建一个CircleActor中即可. 部分代码如下所示: package com.ef.smallstar.unitmap.widget; import android.content.res.Resources; import

  • 详解Android中图片的三级缓存及实例

    详解Android中图片的三级缓存及实例 为什么要使用三级缓存 如今的 Android App 经常会需要网络交互,通过网络获取图片是再正常不过的事了 假如每次启动的时候都从网络拉取图片的话,势必会消耗很多流量.在当前的状况下,对于非wifi用户来说,流量还是很贵的,一个很耗流量的应用,其用户数量级肯定要受到影响 特别是,当我们想要重复浏览一些图片时,如果每一次浏览都需要通过网络获取,流量的浪费可想而知 所以提出三级缓存策略,通过网络.本地.内存三级缓存图片,来减少不必要的网络交互,避免浪费流量

  • 详解Android中Handler的内部实现原理

    本文主要是对Handler和消息循环的实现原理进行源码分析,如果不熟悉Handler可以参见博文<详解Android中Handler的使用方法>,里面对Android为何以引入Handler机制以及如何使用Handler做了讲解. 概括来说,Handler是Android中引入的一种让开发者参与处理线程中消息循环的机制.我们在使用Handler的时候与Message打交道最多,Message是Hanlder机制向开发人员暴露出来的相关类,可以通过Message类完成大部分操作Handler的功

  • 详解Android 中AsyncTask 的使用

    详解Android 中AsyncTask 的使用 1.首先我们来看看AsyncTask 的介绍:   Handler 和 AsyncTask 都是android 中用来实现异步任务处理的方式:其中: Handler 实例向 UI 线程发送消息,完成界面更新, 优点:对整个过程控制的比较精细:         缺点:代码相对臃肿,多个任务同时执行时,不易对线程进行精确的控制: AsyncTask :比Handler 更轻量级一些,适用于简单的异步处理: 优点:简单 | 快捷 | 过程可控:    

  • 详解Android中获取软键盘状态和软键盘高度

    详解Android中获取软键盘状态和软键盘高度 应用场景 在Android应用中有时会需要获取软键盘的状态(即软键盘是显示还是隐藏)和软键盘的高度.这里列举了一些可能的应用场景. 场景一 当软键盘显示时,按下返回键应当是收起软键盘,而不是回退到上一个界面,但部分机型在返回键处理上有bug,按下返回键后,虽然软键盘会自动收起,但不会消费返回事件,导致Activity还会收到这次返回事件,执行回退操作,这时就需要判断,如果软键盘刚刚由显示变为隐藏状态,就不执行回退操作. 场景二 当软键盘弹出后,会将

  • 详解Android中fragment和viewpager的那点事儿

    在之前的博文<Android 中使用 ViewPager实现屏幕页面切换和页面轮播效果>和<详解Android中Fragment的两种创建方式>以及<Android中fragment与activity之间的交互(两种实现方式)>中我们介绍了ViewPager以及Fragment各自的使用场景以及不同的实现方式. 那如果将他们两结合起来,会不会擦出点火花呢,答案是肯定的.之前在介绍ViewPager时,我们实现了多个ImageView的切换,并配合更新导航原点的状态.那我

随机推荐