Android学习笔记 五一 服务Service 上 IntentService

2024-01-05 10:38

本文主要是介绍Android学习笔记 五一 服务Service 上 IntentService,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

分享一下我老师大神的人工智能教程!零基础,通俗易懂!http://blog.csdn.net/jiangjunshow

也欢迎大家转载本篇文章。分享知识,造福人民,实现我们中华民族伟大复兴!

对于需要长期运行,例如播放音乐、长期和服务器的连接,即使已不是屏幕当前的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开发相关文章

           

给我老师的人工智能教程打call!http://blog.csdn.net/jiangjunshow
这里写图片描述

这篇关于Android学习笔记 五一 服务Service 上 IntentService的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/572588

相关文章

Nginx进行平滑升级的实战指南(不中断服务版本更新)

《Nginx进行平滑升级的实战指南(不中断服务版本更新)》Nginx的平滑升级(也称为热升级)是一种在不停止服务的情况下更新Nginx版本或添加模块的方法,这种升级方式确保了服务的高可用性,避免了因升... 目录一.下载并编译新版Nginx1.下载解压2.编译二.替换可执行文件,并平滑升级1.替换可执行文件

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三

RabbitMQ消息总线方式刷新配置服务全过程

《RabbitMQ消息总线方式刷新配置服务全过程》SpringCloudBus通过消息总线与MQ实现微服务配置统一刷新,结合GitWebhooks自动触发更新,避免手动重启,提升效率与可靠性,适用于配... 目录前言介绍环境准备代码示例测试验证总结前言介绍在微服务架构中,为了更方便的向微服务实例广播消息,

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

关于DNS域名解析服务

《关于DNS域名解析服务》:本文主要介绍关于DNS域名解析服务,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录DNS系统的作用及类型DNS使用的协议及端口号DNS系统的分布式数据结构DNS的分布式互联网解析库域名体系结构两种查询方式DNS服务器类型统计构建DNS域

Linux中SSH服务配置的全面指南

《Linux中SSH服务配置的全面指南》作为网络安全工程师,SSH(SecureShell)服务的安全配置是我们日常工作中不可忽视的重要环节,本文将从基础配置到高级安全加固,全面解析SSH服务的各项参... 目录概述基础配置详解端口与监听设置主机密钥配置认证机制强化禁用密码认证禁止root直接登录实现双因素

java向微信服务号发送消息的完整步骤实例

《java向微信服务号发送消息的完整步骤实例》:本文主要介绍java向微信服务号发送消息的相关资料,包括申请测试号获取appID/appsecret、关注公众号获取openID、配置消息模板及代码... 目录步骤1. 申请测试系统2. 公众号账号信息3. 关注测试号二维码4. 消息模板接口5. Java测试

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局