对于需要长期运行,例如播放音乐、长期和服务器的连接,即使已不是屏幕当前的activity仍需要运行的情况,采用服务方式。服务将通过API触发启动或者通过IPC(Interprocess Communication)连接请求触发启动。服务将一直运行直至被关闭,或者内存不足时由系统关闭。一般而言,为了节省电量,服务应进行优化减少CPU的消耗和大量网络通信。服务可用于以下的场景:

1、用户离开activity后,仍需继续工作,例如从网络下载文件,播放音乐

2、无论activity出现(重现)或离开,都需持续工作,例如网络聊天应用 

3、连接网络服务,正在使用一个远程API提供的服务 

4、定时触发的服务,例如Linux中的cron。

在manifest中声明服务

和activity、content provider一样,服务也必须要在AndroidManifest文件中进行声明是<application>中的子节点。例如我们下面第一个service的例子ServiceDownloader。

<application … … > 

     ... ... 

    <service android:name=".ServiceDownloader"> 

</application >

命令模式:IntentService

编写自己的Service将继承Android的Service类,或者Service的子类IntentService。触发Service的方式有两种,一种是发送命令,即这次学习的命令模式,一种绑定服务,与服务之间建立双向的通信渠道。命令模式例子为http远程下载文件的服务。

服务ServiceDownloader

/* 命令模式的服务由client请求服务,服务进行处理,并在完成后关闭服务,client无需关心是否需要结束服务,适合一次性的处理,如本例 */

public class ServiceDownloader extends IntentService{  

    private HttpClient client = null;  

     

    public ServiceDownloader(){ 

        super("ServiceDownloader"); 

    }  

    //client通过startService()请求服务时,如果服务没有开启,则首先执行onCreate(),我们在此进行服务的初始化工作,请注意,onCreate()是在主线程中运行。 

    public void onCreate()
 {  

        super.onCreate(); 

        client = new DefaultHttpClient(); 

    } 

   //如果client发出startService()时,如果服务没有开启,则先开启服务onCreate(),在服务开启后或者如果服务已经开启,将触发onStartCommand(),请注意,这也是在主线程中运行,我们不应用将一些时间长的处理放置此处。一般而言,这里可以根据收到的命令,进行本次服务的初始化处理。原则上,由于是主线程,可进行UI操作,但是好的编程风格,service不处理activity的内容。 

   public int onStartCommand(Intent intent, int flags, int startId) {
 

        return super.onStartCommand(intent, flags, startId);

    } 



   //这是必须override的方法,在收到客户端命令,处理完onStartCommand()后执行,注意onHandlerIntent是在后台线程中运行,应将主要的处理内容放置此处

   protected void onHandleIntent(Intent i) { 

        /*HTTP的例子之前学习过,首先是采用GET的方法获取远程文件。将返回的HTTP存放在responseHandler中,我们写了个私类ByteArrayResponseHandler来处理,检查HTTP的返回值,如果不是200 OK,例如3xx-6xx,则说明出现异常,如成功,将获取的内容存放至文件中。*/ 

        HttpGet getMethod = new HttpGet(i.getData().toString());

        try{ 

            ResponseHandler<byte[]> responseHandler = new ByteArrayResponseHandler(); 

            byte[] responseBody = client.execute(getMethod,responseHandler);

            File output = new File(Environment.getExternalStorageDirectory(), 

            i.getData().getLastPathSegment()); 

            if(output.exists()){ 

                output.delete(); 

            } 

            FileOutputStream fos = new FileOutputStream(output.getPath()); 

            fos.write(responseBody); 

            fos.close(); 

        }catch(Exception e){ 

            Log.e(getClass().getName(),"Exception : " + e.toString()); 

        } 

    } 



    //如果client发出stopService()请求停止服务,或者服务本身通过stopSelf()要求停止服务,都会触发onDestroy(),onDestroy也是在主线程中运行,在此我们应进行停止服务的工作。如果这是正在主线程执行onStartCommand(),则必须要等onStartCommand()的内容执行完,才依次执行onDestroy()的内容。如果这时后台线程onHandleIntent( )正在执行,onDestroy(
)不会自动将后台线程停止,后台线程继续运行,我们必须在onDestroy()的代码中终结后台线程的运行。例如状态检查,或者本地中直接关闭连接,中断通信 

    public void onDestroy()
 {   

        client.getConnectionManager().shutdown(); 

        super.onDestroy(); 

    } 



    //检查返回HTTP Response的返回值,如果是3xx-6xx,不是2xx,则说明出错,例如404,Not Found。

    private class ByteArrayResponseHandler implements ResponseHandler<byte[]>{

        public byte[] handleResponse(HttpResponse response) throws ClientProtocolException, IOException {

            StatusLine statusLine = response.getStatusLine();  

            if(statusLine.getStatusCode() >= 300){  

                throw new HttpResponseException (statusLine.getStatusCode(),statusLine.getReasonPhrase()); 

            } 

            HttpEntity entity = response.getEntity(); 

            if(entity == null) 

                return null; 

            return EntityUtils.toByteArray(entity); 

        } 

    } 

}

命令模式服务的客户端

/*客户端采用命令方式触发服务,由于IntentService在执行完后自动关闭,则只需通过startService( )命令触发即可 */

public class ServiceTest1 extends Activity{  

    … … 

    //调起服务和调起Activity非常相似,都是通过Intent来出传递,通过setData传递参数,在本例是直接http的Uri地址。 

    private void startDownloader(){ 

        Intent intent = new Intent(this,ServiceDownloader.class); 

        intent.setData
(Uri.parse("http://commonsware.com/Android/excerpt.pdf")); 

        startService(intent); 

    } 

   //一般而言命令模式的服务,不需要考虑终止服务。此处只做试验用。注意,终止服务是终止整个服务,会触发服务中的onDestroy( ),如果队列中还有其他命令等等服务处理,将由onDestroy()中的代码停止。因此影响的是所有正在和等待服务处理,而不单是客户端的请求,此需特别注意!!

    private void stopDownloader(){  

        stopService(new Intent(this,ServiceDownloader.class); 

    }  

}

服务和客户端的通信

在上面的例子中,我们希望服务下载完后,能通知客户端。对于命令模式的服务,可采用Messenger的方式,Messenger可以发送消息给activity的Handler,在线程[学习笔记(三一)]中已学习过。

客户端代码如下

public class ServiceTest1 extends Activity{  

    … … 

    private void startDownloader(){ 

        …… 

        intent = new Intent(this,ServiceDownloader.class); 

        intent.setData(Uri.parse("http://commonsware.com/Android/excerpt.pdf"));

        //activity在调起服务时,即startService()或者bindService()都可以携带Messenger作为Intent的extra传递,这样在服务和client之间可通过Messenger传递

        intent.putExtra(ServiceDownloader.EXTRA_MESSAGER, new Messenger(handler)); 

        startService(intent); 

    }  

    //Handler通过handlerMessage()接受消息,运行在主线程,用于处理UI等内容。 

    private Handler handler = new Handler(){ 

        public void handleMessage(Message msg) {  

           super.handleMessage(msg); 

            buttonStart.setEnabled(true); 

            buttonStop.setEnabled(false); 

            switch(msg.arg1){ 

            case Activity.RESULT_OK: 

                Toast.makeText(ServiceTest1.this, "Result : OK " , Toast.LENGTH_LONG).show();

                break; 

            case Activity.RESULT_CANCELED: 

                Toast.makeText(ServiceTest1.this, "Result : Cancel " , Toast.LENGTH_LONG).show();

                break; 

            default:  

                break; 

            } 

        } 

    }; 

}

服务端代码如下:

//避免出现命名重复,将类的命名空间加在前面

    public static final String EXTRA_MESSAGER="com.wei.android.learning.ServiceDownloader.EXTRA_MESSAGER";



    protected void onHandleIntent(Intent i) { 

        int result = Activity.RESULT_CANCELED   

       //下载文件的处理,成功则,设置result = Activity.RESULT_OK; 

        … …    

       //步骤1:从Intent的Extras中获取Messenger

       Bundle extras = i.getExtras();
 

        if(extras != null){ 

            Messenger mesenger = (Messenger)extras.get(EXTRA_MESSAGER); 

           //步骤2:使用Message.obtain()获得一个空的Message对象

            Message msg = Message.obtain( );  

           //步骤3:填充message的信息。  

            msg.arg1 = result; 

          //步骤4:通过Messenger信使将消息发送出去。  

            try{ 

                mesenger.send(msg);  

            }catch(Exception e){ 

                Log.w(getClass().getName(),"Exception Message: " + e.toString());

            } 

        }  

    }

相关链接: 我的Android开发相关文章

android服务Service(上)- IntentService的更多相关文章

  1. Android服务Service具体解释(作用,生命周期,AIDL)系列文章-为什么须要服务呢?

    Android服务Service具体解释(作用,生命周期,AIDL) 近期沉迷于上班,没有时间写博客了.解衣入睡,未眠.随起床写一篇博客压压惊! ##我们android系统为什么须要服务Service ...

  2. Android之Service与IntentService的比较

    Android之Service与IntentService的比较  不知道大家有没有和我一样,以前做项目或者练习的时候一直都是用Service来处理后台耗时操作,却很少注意到还有个IntentServ ...

  3. Android服务——Service

    服务 Service 是一个可以在后台执行长时间运行操作而不使用用户界面的应用组件.服务可由其他应用组件启动,而且即使用户切换到其他应用,服务仍将在后台继续运行. 此外,组件可以绑定到服务,以与之进行 ...

  4. Android服务(Service)研究

    Service是android四大组件之一,没有用户界面,一直在后台运行. 为什么使用Service启动新线程执行耗时任务,而不直接在Activity中启动一个子线程处理? 1.Activity会被用 ...

  5. Android服务Service总结

    转自 http://blog.csdn.net/liuhe688/article/details/6874378 富貴必從勤苦得,男兒須讀五車書.唐.杜甫<柏學士茅屋> 作为程序员的我们, ...

  6. android拾遗——Android之Service与IntentService的比较

    不知道大家有没有和我一样,以前做项目或者练习的时候一直都是用Service来处理后台耗时操作,却很少注意到还有个IntentService,前段时间准备面试的时候看到了一篇关于IntentServic ...

  7. Android服务Service

    安卓Service服务 一    Service简介 Service是运行在后台的,没有界面的,用来处理耗时比较长的.Service不是一个单独的进程,也不是一个单独的线程. Service有两种类型 ...

  8. Android中Service与IntentService的使用比较

    不知道大家有没有和我一样,以前做项目或者练习的时候一直都是用Service来处理后台耗时操作,却很少注意到还有个IntentService,前段时间准备面试的时候看到了一篇关于IntentServic ...

  9. android 服务service开启和关闭

    startService()方法开启一个服务. 服务只会开启一次,如果服务已经创建,并且没有销毁,多次调用startService方法只会执行onStartCommand方法和onStart方法. 服 ...

随机推荐

  1. Python简明教程---学习笔记

    字符双引号括起,数字不括: 分隔符为逗号(,),不能为空格 变量定义时即赋值 采用utf-8编码:#-*-coding:utf-8-*-或者#coding:utf-8 字符串定义:单/双引号括起 %符 ...

  2. jQuery停止事件冒泡

    event.stopPropagation(); 在jQuery中提供了stopPropagation()方法来停止事件冒泡.终止事件在传播过程的捕获.目标处理或起泡阶段进一步传播.调用该方法后,该节 ...

  3. [问题解决] /home目录占用率100%

    今天发现一个比较奇怪的现象,linux系统下一个目录挂在存储下,df -Th 显示该目录使用率100%, du 该目录显示只用了2%, 后来发现是由于进程占用了被删掉的文件空间导致.举例如下: [ro ...

  4. ASP 代码当前记录集不支持更新问题的解决办法。

    错误类型: ADODB.Recordset (0x800A0CB3) 当前记录集不支持更新.这可能是提供程序的限制,也可能是选定锁定类型的限制. /Model/manage/Admin_Admin.a ...

  5. java线程管理

    java线程管理 参见: http://harmony.apache.org/subcomponents/drlvm/TM.html 1. 修订历史 2. 关于本文档 2.1. 目的 2.2. 面向的 ...

  6. windows、linux创建子进程

    在windows下创建子进程较常用到的API就是CreateProcess,可以通过以下的方式启动一个新进程: STARTUPINFO si = {0};                 PROCES ...

  7. linux中ctrl+z、ctrl+d和ctrl+c的区别

    ctrl+c和ctrl+z都是中断命令,但是他们的作用却不一样.ctrl+c是强制中断程序的执行,而ctrl+z的是将任务中断,但是此任务并没有结束,他仍然在进程中他只是维持挂起的状态,用户可以使用f ...

  8. win下vm10+mac os 10.9安装遇到问题

    在windows 8下安装vm10.0.0+mac os 10.9遇到问题记录例如以下: 一.因为之前我装的vm9+mac os 10.7: 二.准备安装mac os 10.9,把vm9换成vm10: ...

  9. 从头开始-02.C语言基础

    变量的内存分析: #include <stdio.h> int main() { //内存地址由大到小 int a=10; int b=20; //&是一个地址运算符,取得变量的地 ...

  10. OC语法7——内存管理之@property参数

    @property的参数: 我们已经知道为了给开发者提供便捷,OC提供了@porperty关键字,它可以自动生成属性的set和get方法. 但是我们又知道,在OC中还面临者对象内存管理的问题,而且我们 ...