Android——Service介绍与例子

    官方定义:Service(服务)是一个没有用户界面的在后台运行执行耗时操作的应用组件。其他应用组件能够启动Service,并且当用户切换到另外的应用场景,Service将持续在后台运行。另外,一个组件能够绑定到一个service与之交互(IPC机制),例如,一个service可能会处理网络操作,播放音乐,操作文件I/O或者与内容提供者(content provider)交互,所有这些活动都是在后台进行。

Service有两种状态,“启动的”和“绑定”

 

其中涉及到了Service的两种状态:(Started)、(Bound

  1. Started:

    通过startService()启动的服务处于“启动的”状态,一旦启动,service就在后台运行,即使启动它的应用组件已经被销毁了。通常started状态的service执行单任务并且不返回任何结果给启动者。比如当下载或上传一个文件,当这项操作完成时,service应该停止它本身。

  2. Bound:

    “绑定”状态的service,通过调用bindService()来启动,一个绑定的service提供一个允许组件与service交互的接口,可以发送请求、获取返回结果,还可以通过夸进程通信来交互(IPC)。绑定的service只有当应用组件绑定后才能运行,多个组件可以绑定一个service,当调用unbind()方法时,这个service就会被销毁了。

     

    注意:

    a.一个服务在进程中的主线程运行——一个服务不会创建自己的线程,也不会在另外的进程运行。

    b.还有就是不要把Service理解成线程,你可以把 Service 想象成一种消息服务,而你可以在任何有 Context 的地方调用 Context.startServiceContext.stopServiceContext.bindServiceContext.unbindService,来控制它,你也可以在 Service 里注册 BroadcastReceiver,在其他地方通过发送 broadcast 来控制它,当然这些都是 Thread 做不到的。举个简单的小例子:如果你的 Thread 需要不停地隔一段时间就要连接服务器做某种同步的话,该 Thread 需要在 Activity 没有start的时候也在运行。这个时候当你 start 一个 Activity 就没有办法在该 Activity 里面控制之前创建的 Thread。因此你便需要创建并启动一个 Service ,在 Service 里面创建、运行并控制该 Thread,这样便解决了该问题(因为任何 Activity 都可以控制同一 Service,而系统也只会创建一个对应 Service 的实例)。

具体例子:在布局文件中添加两个按钮(启动服务,停止服务),添加点击事件(具体findById获取,再添加监听事件)

创建一个Service

package com.example.test_myservice_demo01;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;

public class MyService extends Service {
    public MyService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        throw new UnsupportedOperationException("Not yet implemented");
    }

    @Override
    public void onCreate() {
        super.onCreate();
        System.out.println("创建服务");
    }

    @Override
    public int onStartCommand(Intent intent,int flags, int startId) {
        System.out.println("启动服务...");      //这里实现服务的核心业务
        for (int i=0;i<50;i++){
            System.out.println("i="+i);
        }
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onDestroy() {
        System.out.println("停止服务");
        super.onDestroy();
    }
}

 

MainActivity文件

...

public void onClick(View v) {
    switch (v.getId()){
        case R.id.button_start:
            Intent startIntent = new Intent(this,MyService.class);
            startService(startIntent);     //启动服务
            break;
        case R.id.button_stop:
            Intent stopIntent = new Intent(this,MyService.class);
            stopService(stopIntent);   //停止服务
            break;

       default:
            break;
    }
}

 

manifest清单文件中可以看到:

<service
    android:name=".MyService"
    android:enabled="true"
    android:exported="true" />

实验结果:如果不点击停止服务,就会一直在运行状态,有什么办法可以能让服务停止下来,只需要在MyService的任何位置调用stopSelf()就能停止服务。

 

 04-17 06:12:36.755 2909-2909/com.example.test_myservice_demo01 I/System.out: 创建服务
04-17 06:12:36.755 2909-2909/com.example.test_myservice_demo01 I/System.out: 启动服务...
04-17 06:12:36.755 2909-2909/com.example.test_myservice_demo01 I/System.out: i=0
04-17 06:12:36.756 2909-2909/com.example.test_myservice_demo01 I/System.out: i=1
04-17 06:12:36.756 2909-2909/com.example.test_myservice_demo01 I/System.out: i=2
04-17 06:12:36.756 2909-2909/com.example.test_myservice_demo01 I/System.out: i=3

...
04-17 06:12:36.757 2909-2909/com.example.test_myservice_demo01 I/System.out: i=47
04-17 06:12:36.757 2909-2909/com.example.test_myservice_demo01 I/System.out: i=48
04-17 06:12:36.757 2909-2909/com.example.test_myservice_demo01 I/System.out: i=49

点击停止按钮后之后
04-17 06:12:43.282 2909-2909/com.example.test_myservice_demo01 I/System.out: 停止服务

...

 

IntentService服务

这是一个Service的子类,IntentService使用队列的方式将请求的Intent加入队列,然后开启一个worker thread(线程)来处理队列中的Intent,对于异步的startService请求,IntentService会处理完成一个之后再处理第二个,每一个请求都会在一个单独的worker thread中处理,不会阻塞应用程序的主线程,这里就给我们提供了一个思路,如果有耗时的操作与其在Service里面开启新线程还不如使用IntentService来处理耗时操作。而在一般的继承Service里面如果要进行耗时操作就必须另开线程,但是使用IntentService就可以直接在里面进行耗时操作,因为默认实现了一个worker thread。对于异步的startService请求,IntentService会处理完成一个之后再处理第二个。

 

例子:

Service

package com.example.test_myservice_demo01;
import android.app.IntentService;
import android.content.Intent;
/**
 * An {@link IntentService} subclass for handling asynchronous task requests in
 * a service on a separate handler thread.
 * <p>
 * TODO: Customize class - update intent actions and extra parameters.
 * 内部类只有一个工作线程来完成耗时操作,只需要实现HandleIntent方法就行
 */
public class MyIntentService extends IntentService {
    public MyIntentService() {
        super("MyIntentService");
    }

    @Override
    protected void onHandleIntent(Intent intent) {
        System.out.println(intent.getStringExtra("info"));
        for (int i =0;i<15;i++){
            System.out.println("onHandleIntent-"+i+"-"+Thread.currentThread().getName());
            try {
                Thread.sleep(500);  //模拟延迟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
//            if (i==10){
//                stopSelf(); //如果添加,便会停止服务
//                break;
//            }
        }
    }
}

MainActivity同上加上一个点击按钮

case R.id.button_3:
    Intent startIntentService = new Intent(this,MyIntentService.class);
    startIntentService.putExtra("info","这个是 我的第一个IntentService");
    startService(startIntentService);
    break;

-----------------分割线---------------------------

 

Bound还有就是绑定服务、解绑

不同于以上服务,在MainActivity中加按钮点击事件(bindunbind

case R.id.button_bind:
    Intent intent = new Intent(this,MyBoundService.class);
    //异步
    bindService(intent,connection, Context.BIND_AUTO_CREATE);            //绑定服务
    break;
case R.id.button_unbind:
    unbindService(connection);                                  //解除服务
    break;

还需要:

//绑定服务的连接回调方法
private ServiceConnection connection = new ServiceConnection() {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        //绑定成功后回调方法
        downloadBinder = (MyBoundService.DownloadBinder) service;
        downloadBinder.startDownload();
        downloadBinder.getProgress();
    }
    @Override
    public void onServiceDisconnected(ComponentName name) {
        //服务异常调用
    }
};

 

Service类:

package com.example.test_myservice_demo01;
import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;
public class MyBoundService extends Service {
    private DownloadBinder mBinder = new DownloadBinder();
    class DownloadBinder extends Binder{
        public void startDownload(){
            Log.d("MyService","startDownload executed");
        }
        public int getProgress(){
            Log.d("MyService","getProgress executed");
         return 0;
        }
    }
    public MyBoundService() {
    }
    @Override
    public void onCreate() {
        super.onCreate();
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        return mBinder;
//        throw new UnsupportedOperationException("Not yet implemented");
    }

    @Override
    public boolean onUnbind(Intent intent) {
        System.out.println("已经解除绑定");
        return super.onUnbind(intent);
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
    }
}

实验结果:

绑定:

04-17 06:50:08.346 23840-23840/com.example.test_myservice_demo01 D/MyService: startDownload executed

04-17 06:50:08.346 23840-23840/com.example.test_myservice_demo01 D/MyService: getProgress executed

解绑:

04-17 06:50:58.931 23840-23840/com.example.test_myservice_demo01 I/System.out: 已经解除绑定

 

转载于:https://www.cnblogs.com/llw1996/p/6722814.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值