zoukankan      html  css  js  c++  java
  • Android开发Service

    一、简介

    service是android应用4种组件之一,是没有界面的在后台运行的服务。

    service有两种形式

    1. 启动的(Started)

    通过应用组件的接口startService()启动,服务启动后独立运行。

    2. 绑定的(Bound)

    通过应用组件的接口bindService()启动,提供CS方式的进程间通讯,在所有绑定的组件解除绑定后,服务退出。

    这两种形式可以单独也可以同时存在于一个service上。

    注意:

    默认service运行在主线程中,并没有独立线程、进程专门运行service。为了防止ANR,应专门启动一个线程运行service

    对于有界面的应用,且后台只在界面运行时运行的,建议使用Thread或者AsyncTaskHandlerThread

    为了防止其他应用访问服务,可以在AndroidManifest.xml定义service为private

    二、service的生命周期

    当内存不足时,service会被杀死,并且运行时间越久越容易被杀死。当service运行在前台,则几乎不被杀死。

    三、使用

    1. 声明服务

    在AndroidManfest.xml中添加

    <manifest ... >
      ...
      <application ... >
          <service android:enabled=["true" | "false"]      // 是否被系统枚举?
             android:exported=["true" | "false"]           // false表示只有相同userID的应用可以访问。
             android:icon="drawable resource"
             android:label="string resource"
             android:name="string"                         // 唯一的必须属性
             android:permission="string"                   // 其他组件调用startService、bindService、stopService时需要添加该权限
             android:process="string" >                    // 运行的进程名,":"开始表示应用私有。
                  <intent-filter>
                  . . .
                 <meta-data>
                  . . .
          </service>
          ...
      </application>
    </manifest>

    2、扩展IntentService

    IntentService是Service的封装,它把多个Intent放在一个队列里,一个一个处理。应用开发者只需要实现构造函数及onHandleIntent()即可。

    或者使用扩展Service。

    3、扩展Service

    当需要多线程处理Intent来加强实时性时,

    public class HelloService extends Service {
      private Looper mServiceLooper;
      private ServiceHandler mServiceHandler;
    
      // Handler that receives messages from the thread
      private final class ServiceHandler extends Handler {
          public ServiceHandler(Looper looper) {
              super(looper);
          }
          @Override
          public void handleMessage(Message msg) {
              // Normally we would do some work here, like download a file.
              // For our sample, we just sleep for 5 seconds.
              long endTime = System.currentTimeMillis() + 5*1000;
              while (System.currentTimeMillis() < endTime) {
                  synchronized (this) {
                      try {
                          wait(endTime - System.currentTimeMillis());
                      } catch (Exception e) {
                      }
                  }
              }
              // Stop the service using the startId, so that we don't stop
              // the service in the middle of handling another job
              stopSelf(msg.arg1);
          }
      }
    
      @Override
      public void onCreate() {
        // Start up the thread running the service.  Note that we create a
        // separate thread because the service normally runs in the process's
        // main thread, which we don't want to block.  We also make it
        // background priority so CPU-intensive work will not disrupt our UI.
        HandlerThread thread = new HandlerThread("ServiceStartArguments",
                Process.THREAD_PRIORITY_BACKGROUND);
        thread.start();
    
        // Get the HandlerThread's Looper and use it for our Handler
        mServiceLooper = thread.getLooper();
        mServiceHandler = new ServiceHandler(mServiceLooper);
      }
    
      @Override
      public int onStartCommand(Intent intent, int flags, int startId) {
          Toast.makeText(this, "service starting", Toast.LENGTH_SHORT).show();
    
          // For each start request, send a message to start a job and deliver the
          // start ID so we know which request we're stopping when we finish the job
          Message msg = mServiceHandler.obtainMessage();
          msg.arg1 = startId;
          mServiceHandler.sendMessage(msg);
    
          // If we get killed, after returning from here, restart
          return START_STICKY;
      }
    
      @Override
      public IBinder onBind(Intent intent) {
          // We don't provide binding, so return null
          return null;
      }
    
      @Override
      public void onDestroy() {
        Toast.makeText(this, "service done", Toast.LENGTH_SHORT).show();
      }
    }

    其中onStartCommand() 返回以下3个值之一

    START_NOT_STICKY:表示服务被杀死后不再重新生成
    START_STICKY:服务被杀死后重启,但 Intent为空,除非有未完成的Intent.
    START_REDELIVER_INTENT:服务被杀死后重启,且发送最后一次的Intent。

    4.启动服务

    Intent intent = new Intent(this, HelloService.class);
    startService(intent);

    5. 停止服务

    服务自己stopService()

    或者其他组件调用stopService()

    四、前台Service

    前台服务通过在服务生命周期内有不可消除的通知来达到。

    1. 启用前台服务

    Notification notification = new Notification(R.drawable.icon, getText(R.string.ticker_text),
            System.currentTimeMillis());
    Intent notificationIntent = new Intent(this, ExampleActivity.class);
    PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);
    notification.setLatestEventInfo(this, getText(R.string.notification_title),
            getText(R.string.notification_message), pendingIntent);
    startForeground(ONGOING_NOTIFICATION, notification);

    2. 停止前台服务

    通过调用stopForeground(true)

  • 相关阅读:
    Linux 的文件软链接如何删除
    mysql-xtrabackup备份sh: xtrabackup_56: command not found与error while loading shared libraries: libssl.so.6: cannot open shared object file: No such file or directory
    How To Upgrade ASMLib Kernel Driver as Part of Kernel Upgrade? (文档 ID 1391807.1)
    [trouble] error connecting to master 'repl@192.168.1.107:3306'
    Troubleshooting 10g and 11.1 Clusterware Reboots (文档 ID 265769.1)
    "Last_IO_Error: Fatal error: The slave I/O thread stops because master and slave have equal MySQL server UUIDs
    RMAN-06900 RMAN-06901 ORA-19921
    一则ORACLE进程都在但是无法进入实例的问题
    VirtualBox下Win7下CPU高占用的一次故障解决
    netcore之mysql中文乱码问题解决记录
  • 原文地址:https://www.cnblogs.com/thinkfeed/p/2634891.html
Copyright © 2011-2022 走看看