服务可用在一下情景:

1,用户离开activity后,仍需要继续工作,例如从网络下载文件,播放音乐.
2,无论activity出现或离开,都需要持续工作,例如网络聊天应用.
3,连接网络服务,正在使用一个远程API提供的服务.
4,定时触发的任务

1.因为IntentService是Service子类,所以也需要在manifest中声明服务

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.intentservicedemo"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="21" />
    <uses-permission android:name="android.permission.INTERNET"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <service android:name="com.example.intentservicedemo.ServiceDownloader"></service>
    </application>
</manifest>

2,新建 ServiceDownloader 继承自 IntentService

public class ServiceDownloader extends IntentService {
       //避免出现命名重复,将类的命名空间加在前面
    public static final String EXTRA_MESSAGER="com.wei.android.learning.ServiceDownloader.EXTRA_MESSAGER";
    public ServiceDownloader() {
        super("ServiceDownloader");
    }
    //命令模式的服务由client请求服务,
    private HttpClient client=null;
    int result ;
    private Intent mIntent;
    
    //client通过startService请求服务时,如果没有开启,则会首先执行onCreate(),在这里做一些初始化的工作,onCreate是在主线程中运行
    @Override
    public void onCreate() {
        super.onCreate();
        client=new DefaultHttpClient();
    }
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        return super.onStartCommand(intent, flags, startId);
        
    }
    //再收到客户端命令,处理完onStartCommand()后执行,onHandleIntent()是在后台线程中运行
    @Override
    protected void onHandleIntent(Intent i) {
        this.mIntent=i;
        result = Activity.RESULT_CANCELED; 
        HttpGet get=new HttpGet(i.getData().toString());
        ResponseHandler<byte[]> responseHandler =new ByteArrayResponseHandler();
        try {
            byte[] responseByte=client.execute(get,responseHandler);
            File output=new File(Environment.getExternalStorageDirectory(),i.getData().getLastPathSegment());
            if(output.exists()){
                output.delete();
            }
            FileOutputStream fos=new FileOutputStream(output.getPath());
            fos.write(responseByte);
            fos.close();
            result=Activity.RESULT_OK;
        } catch (ClientProtocolException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        Bundle extras=i.getExtras();
        if(extras!=null){
            Messenger messenger =(Messenger) extras.get(EXTRA_MESSAGER);
            Message msg=Message.obtain();
            msg.arg1=result;
            try {
                messenger.send(msg);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
    }
    
    @Override
    public void onDestroy() {
        Bundle extras=mIntent.getExtras();
        if(extras!=null){
            Messenger messenger =(Messenger) extras.get(EXTRA_MESSAGER);
            Message msg=Message.obtain();
            msg.arg1=result;
            try {
                messenger.send(msg);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        client.getConnectionManager().shutdown();
        super.onDestroy();
    }
    
    private class ByteArrayResponseHandler implements ResponseHandler<byte[]>{
        @Override
        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);
        }
    }
}

3,在客户端调用服务,开启下载任务,并且在下载完成时,可以收到下载完成的消息
public class MainActivity extends ActionBarActivity {
    
    Button btn_start;
    Button btn_stop;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        btn_start=(Button) findViewById(R.id.btn_start);
        btn_stop=(Button) findViewById(R.id.btn_stop);
        btn_start.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                startDownLoader();
            }
        });
        btn_stop.setOnClickListener(new OnClickListener() {
            
            @Override
            public void onClick(View arg0) {
                stopDownLoader();
            }
        });
    }
    
    private void startDownLoader(){
        Intent intent=new Intent(this, ServiceDownloader.class);
        intent.setData(Uri.parse("http://commonsware.com/Android/excerpt.pdf")); 
        intent.putExtra(ServiceDownloader.EXTRA_MESSAGER,new Messenger(mh));
        startService(intent);
    }
    
    private Handler mh=new Handler(){
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            //下载完成后,将取消服务按钮失效
            switch (msg.arg1) {
            case Activity.RESULT_OK:
                Toast.makeText(MainActivity.this, "Result : OK " , Toast.LENGTH_LONG).show();
                break;
            case Activity.RESULT_CANCELED:
                Toast.makeText(MainActivity.this, "Result : Cancel " , Toast.LENGTH_LONG).show();
                break;
            default:
                break;
            }
        }
    };
    private void stopDownLoader(){
        stopService(new Intent(this, ServiceDownloader.class));
    }
    
}

4,activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context="com.example.intentservicedemo.MainActivity" >
    <Button 
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="start"
        android:id="@+id/btn_start"
        />
    <Button 
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="stop"
        android:id="@+id/btn_stop"
        />
</LinearLayout>


android命令模式IntentService 远程下载文件的更多相关文章

  1. 通过cmd命令到ftp上下载文件

    通过cmd命令到ftp上下载文件 点击"开始"菜单.然后输入"cmd"点"enter"键,出现cmd命令执行框 2 输入"ftp& ...

  2. PHP实现远程下载文件到本地

    PHP实现远程下载文件到本地 投稿:hebedich 字体:[增加 减小] 类型:转载   经常写采集器发布接口需要使用到远程附件的功能,所以自己写了一个PHP远程下载文件到本地的函数,一般情况下已经 ...

  3. PHP CURL实现远程下载文件到本地

    <?php //$result=httpcopy('http://www.phpernote.com/image/logo.gif'); echo '<pre>';print_r($ ...

  4. Android开发 ---从互联网上下载文件,回调函数,图片压缩、倒转

     Android开发 ---从互联网上下载文件,回调函数,图片压缩.倒转 效果图: 描述: 当点击“下载网络图像”按钮时,系统会将图二中的照片在互联网上找到,并显示在图像框中 注意:这个例子并没有将图 ...

  5. linux远程下载文件 的两种方法之 ftp命令和scp命令

    ftp命令: 服务器有安装ftp Server,另外一台linux可以使用ftp的client程序来进行文件的拷贝读取和下载. 1. 连接ftp服务器  格式:ftp [hostname| ip-ad ...

  6. 【liunx命令】上传下载文件的方法

    scp   帮助命令: man scp   scp功能: 下载远程文件或者目录到本地, 如果想上传或者想下载目录,最好的办法是采用tar压缩一下,是最明智的选择.   从远程主机 下载东西到 本地电脑 ...

  7. 批量执行(Linux命令,上传/下载文件)

    前言: 每个公司的网络环境大都划分 办公网络.线上网络,之所以划分的主要原因是为了保证线上操作安全: 对于外部用户而言也只能访问线上网络的特定开放端口,那么是什么控制了用户访问线上网络的呢? 防火墙过 ...

  8. Linux学习笔记:使用ftp命令上传和下载文件

    Linux中如何使用ftp命令,包括如何连接ftp服务器,上传or下载文件以及创建文件夹.虽然现在有很多ftp桌面应用(例如:FlashFXP),但是在服务器.SSH.远程会话中掌握命令行ftp的使用 ...

  9. python3 实现堡垒机功能(并发执行命令及上传下载文件)

    转载请注明出处,欢迎提出宝贵意见,谢谢! 功能介绍: 1.主机分组 登录后显示分组主机及主机数量 选择主机组后显示该主机组下所有主机信息,主机名及IP显示输入选择:1.执行命令利用线程并发组内所有主机 ...

随机推荐

  1. 使用RMAN对数据文件进行恢复

    (1)备份数据库 在使用RMAN进行数据库恢复之前,先用RMAN进行全库备份 [oracle@redhat6 ~]$ rman target / Recovery Manager: Release : ...

  2. cookie与session的区别,你真的明白吗?

    当我们访问网页时,http是属于无状态的,为什么呢?接下来由我慢慢讲解,在cookie的到来之前,你第一次访问页面的时候和最后一次访问页面服务器是不知道的,不知道那一次访问的页面是你.当用户登录的时候 ...

  3. window/linux下获取文件MD5

    MD5消息摘要算法(英语: MD5 Message-Digest Algorithm), 主要用于确保信息传输过程的一致性校验.   首先介绍两个工具: window: WinMD5Free Linu ...

  4. 20181009noip HZ EZ 两校联考trade(优先队列,贪心)

    题面戳这里 思路: 裸的,贪心... 考场上写了一个数据分治(70ptsDP,30pts线段树优化贪心,GG了后30分) 这道题其实很简单的 我们看图: 我们在A时刻买一个东西,在B时刻卖出去,我们可 ...

  5. ETO的公开赛T1《矿脉开采》题解(正解)(by Zenurik)

    作为T1,当然是越水越好啦qwq 显然经目测可得,那个所谓的质量评级根本就没卵用,可以直接\(W_i = W_i^{V_i}\)累积到利润里面. 这样,本问题显然是一个"子集和"问 ...

  6. [洛谷P1390]公约数的和·莫比乌斯反演

    公约数的和 传送门 分析 这道题很显然答案为 \[Ans=\sum_{i=1}^n\sum_{j=i+1}^n (i,j)\] //其中\((i,j)\)意味\(gcd(i,j)\) 这样做起来很烦, ...

  7. 独木舟(51NOD 1432 )

    n个人,已知每个人体重.独木舟承重固定,每只独木舟最多坐两个人,可以坐一个人或者两个人.显然要求总重量不超过独木舟承重,假设每个人体重也不超过独木舟承重,问最少需要几只独木舟? Input 第一行包含 ...

  8. java程序员所必须掌握的内容

    以下内容有待细化,并用于考察员工的水平! 从低的往高级的说. 初级 1.掌握java语法 oop+io+网络 2.基本的数据结构 3.基本的算法-例如排序,合并 4.基本的几个javaee框架 spr ...

  9. Axure RP Extension for Chrome安装

    Axure RP Extension for Chrome安装 Axure RP Extension for Chrome是一款谷歌插件,主要可以用来查看原型文件.以前安装插件的时候总是找半天资源,很 ...

  10. 【PHP】统计问卷调查结果的选项票数和百分比

    遇到问题: 有以下数组,每一条记录是用户的每一条问卷题目的回答情况,q_id是问题id,o_id是选项id.需要统计每一个选项被选择的次数和每个选项占该问题的百分比.如问题1的选项有A和B,一个用户选 ...