热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android基于IntentService的文件下载的示例代码

文件下载这种事情是很耗时的。之前使用AsyncTask这样的异步类来做下载,然后切到后台就被干掉。所以打算试试Service。(不过按目前那

文件下载这种事情是很耗时的。之前使用AsyncTask这样的异步类来做下载,然后切到后台就被干掉。所以打算试试Service。(不过按目前那些系统的尿性,其实Service也分分钟被干掉)

不过,这里并不是直接使用Service类,而是使用的是继承自Service的IntentService。

这个东西有三大好处:

1.他有个任务队列;
2.任务队列执行完后会自动停止;
3.他会起一个独立的线程,耗时操作不会影响你app的主线程。

这么自动化的东西简直省心。

话不多说,开始撸代码。

首先,要建个应用,主文件如下(布局什么的代码就不贴了):

package net.codepig.servicedownloaderdemo;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

public class MainActivity extends AppCompatActivity {
  private String _url="http://www.boosj.com/apk/boosjDance.apk";
  private EditText urlText;
  private Button goBtn;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    goBtn=(Button) findViewById(R.id.goBtn);
    urlText=(EditText) findViewById(R.id.urlText);
    urlText.setText(_url);
    goBtn.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View view) {
        _url=urlText.getText().toString();
        //start download
        start_service();
      }
    });
  }

  public void start_service(){
    //等会再填
  }
}

以上代码不重要,嗯。

接下来是重点。创建一个IntentService 类,然后重写他的onHandleIntent 。

需要执行的任务就写在onHandleIntent 里

这里先用Thread.sleep模拟一下耗时任务,试跑一下就可以看到主app关闭后service还在跑,跑完后就自己Destroy了。

package net.codepig.servicedownloaderdemo;

import android.app.IntentService;
import android.content.Intent;

/**
 * 下载服务
 * Created by QZD on 2017/9/20.
 */

public class DownLoadService extends IntentService {
  public DownLoadService() {
    super("DownLoadService");//这就是个name
  }

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

  protected void onHandleIntent(Intent intent) {
    Bundle bundle = intent.getExtras();
    String downloadUrl = bundle.getString("download_url");

    Log.d(TAG,"下载启动:"+downloadUrl);
    Thread.sleep(1_000);
    int count=0;
    while(count<20){
      count++;
      Log.d(TAG,"下载运行中--"+count);
      Thread.sleep(1000);
    }
    Log.d(TAG,"下载结束");
  }

  @Override
  public void onDestroy() {
    Log.d(TAG, "onDestroy");
    super.onDestroy();
  }
}

通过Intent接收任务,这里在MainActivity中通过startService启动服务

public void start_service(){
    Intent intent=new Intent(this,DownLoadService.class);
    intent.putExtra("download_url",_url);
    startService(intent);
  }

当然,AndroidManifest.xml里也得注册上

代码如下:



接下来我们看看怎么下载文件

首先别忘了添加权限:





添加downloadFile方法管理下载。

下载相关说明都在注释里。

  /**
   * 文件下载
   * @param downloadUrl
   * @param file
   */
  private void downloadFile(String downloadUrl, File file){
    FileOutputStream _outputStream;//文件输出流
    try {
      _outputStream = new FileOutputStream(file);
    } catch (FileNotFoundException e) {
      Log.e(TAG, "找不到目录!");
      e.printStackTrace();
      return;
    }
    InputStream _inputStream = null;//文件输入流
    try {
      URL url = new URL(downloadUrl);
      HttpURLConnection _downLoadCon = (HttpURLConnection) url.openConnection();
      _downLoadCon.setRequestMethod("GET");
      fileLength = Integer.valueOf(_downLoadCon.getHeaderField("Content-Length"));//文件大小
      _inputStream = _downLoadCon.getInputStream();
      int respOndCode= _downLoadCon.getResponseCode();//服务器返回的响应码
      if (respOndCode== 200) {
        byte[] buffer = new byte[1024*8];// 数据块,等下把读取到的数据储存在这个数组,这个东西的大小看需要定,不要太小。
        int len;
        while ((len = _inputStream.read(buffer)) != -1) {
          _outputStream.write(buffer, 0, len);
          downloadLength = downloadLength + len;
          Log.d(TAG, downloadLength + "/" + fileLength );
        }
      } else {
        Log.d(TAG, "respondCode:" + respondCode);
      }
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      try {//别忘了关闭流
        if (_outputStream != null) {
          _outputStream.close();
        }
        if (_inputStream != null) {
          _inputStream.close();
        }
      } catch (IOException e) {
        e.printStackTrace();
      }
    }
  }

放入onHandleIntent执行:

protected void onHandleIntent(Intent intent) {
    try {
      Bundle bundle = intent.getExtras();
      String downloadUrl = bundle.getString("download_url");

      File dirs = new File(Environment.getExternalStorageDirectory().getAbsolutePath() + "/Download");//文件保存地址
      if (!dirs.exists()) {// 检查文件夹是否存在,不存在则创建
        dirs.mkdir();
      }

      File file = new File(dirs, "boosj.apk");//输出文件名
      Log.d(TAG,"下载启动:"+downloadUrl+" --to-- "+ file.getPath());
      // 开始下载
      downloadFile(downloadUrl, file);
      // 下载结束
      Log.d(TAG,"下载结束");
    } catch (Exception e) {
      e.printStackTrace();
    }
  }

跑一下,嗯,默默的下载完了。

但是,作为一个负责的app,当然要给用户反馈,所以我们要显示一下进度。

我们用Notification来显示进度。(需要注意的是,如果有必要调用主UI线程来显示进度的话,要充分考虑到Service运行过程中,你的app未必是一直活动着的,可能早就destroy了。)(当然用绑定来启动service的另说,那是另一种使用场景。)
下载前(也就是执行downloadFile方法前)先创建并对通知进行相关设置。

这里使用了NotificationCompat.Builder()这个方法。如果不考虑旧版本的兼容,可以使用Notification.Builder()方法。

private NotificationCompat.Builder builder;
private NotificationManager manager;
public void initNotification(){
    builder = new NotificationCompat.Builder(this);
    builder.setSmallIcon(R.mipmap.ic_launcher).setContentTitle("下载文件").setContentText("下载中……");//图标、标题、内容这三个设置是必须要有的。
    manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
  }

然后使用NotificationManager.notify()方法将通知发送给系统。需要更新的话再次notify()给同一个ID的通知,如果该通知已存在则会更新,不存在就新建。

private int _notificatiOnID= 1024;//嗯,这是一个十分绅士的ID
manager.notify(_notificationID,builder.build());

为了显示进度,使用handler和Runnable来定时刷新,并通过setProgress方法显示进度条。

private Handler handler = new Handler();
private Runnable run = new Runnable() {
    public void run() {
      int _pec=(int) (downloadLength*100 / fileLength);
      builder.setContentText("下载中……"+_pec+"%");
      builder.setProgress(100, _pec, false);//显示进度条,参数分别是最大值、当前值、是否显示具体进度(false显示具体进度,true就只显示一个滚动色带)
      manager.notify(_notificationID,builder.build());
      handler.postDelayed(run, 1000);
    }
  };

完事了以后如果需要清除通知可以使用manager.cancelAll();或者manager.cancel(int );

完整代码再来一遍

package net.codepig.servicedownloaderdemo;

import android.app.IntentService;
import android.app.NotificationManager;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.support.v4.app.NotificationCompat;
import android.util.Log;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;

/**
 * 下载服务
 * Created by QZD on 2017/9/20.
 */

public class DownLoadService extends IntentService {
  private final String TAG="LOGCAT";
  private int fileLength, downloadLength;//文件大小
  private Handler handler = new Handler();
  private NotificationCompat.Builder builder;
  private NotificationManager manager;
  private int _notificatiOnID= 1024;
  public DownLoadService() {
    super("DownLoadService");//这就是个name
  }

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

  protected void onHandleIntent(Intent intent) {
    try {
      initNotification();

      Bundle bundle = intent.getExtras();
      String downloadUrl = bundle.getString("download_url");

      File dirs = new File(Environment.getExternalStorageDirectory().getAbsolutePath() + "/Download");//文件保存地址
      if (!dirs.exists()) {// 检查文件夹是否存在,不存在则创建
        dirs.mkdir();
      }

      File file = new File(dirs, "boosj.apk");//输出文件名
      Log.d(TAG,"下载启动:"+downloadUrl+" --to-- "+ file.getPath());
      manager.notify(_notificationID,builder.build());
      // 开始下载
      downloadFile(downloadUrl, file);
      // 下载结束
      builder.setProgress(0,0,false);//移除进度条
      builder.setContentText("下载结束");
      manager.notify(_notificationID,builder.build());
//      manager.cancelAll();
//      manager.cancel(_notificationID);

      // 广播下载完成事件,通过广播调起对文件的处理。(就不多说了,在实际需要的地方接收广播就好了。)
      Intent sendIntent = new Intent("downloadComplete");
      sendIntent.putExtra("downloadFile", file.getPath());
      sendBroadcast(sendIntent);
      Log.d(TAG,"下载结束");
    } catch (Exception e) {
      e.printStackTrace();
    }
  }

  /**
   * 文件下载
   * @param downloadUrl
   * @param file
   */
  private void downloadFile(String downloadUrl, File file){
    FileOutputStream _outputStream;//文件输出流
    try {
      _outputStream = new FileOutputStream(file);
    } catch (FileNotFoundException e) {
      Log.e(TAG, "找不到目录!");
      e.printStackTrace();
      return;
    }
    InputStream _inputStream = null;//文件输入流
    try {
      URL url = new URL(downloadUrl);
      HttpURLConnection _downLoadCon = (HttpURLConnection) url.openConnection();
      _downLoadCon.setRequestMethod("GET");
      fileLength = Integer.valueOf(_downLoadCon.getHeaderField("Content-Length"));//文件大小
      _inputStream = _downLoadCon.getInputStream();
      int respOndCode= _downLoadCon.getResponseCode();//服务器返回的响应码
      if (respOndCode== 200) {
        handler.post(run);//更新下载进度
        byte[] buffer = new byte[1024*8];// 数据块,等下把读取到的数据储存在这个数组,这个东西的大小看需要定,不要太小。
        int len;
        while ((len = _inputStream.read(buffer)) != -1) {
          _outputStream.write(buffer, 0, len);
          downloadLength = downloadLength + len;
//          Log.d(TAG, downloadLength + "/" + fileLength );
        }
      } else {
        Log.d(TAG, "respondCode:" + respondCode);
      }
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      try {//别忘了关闭流
        if (_outputStream != null) {
          _outputStream.close();
        }
        if (_inputStream != null) {
          _inputStream.close();
        }
      } catch (IOException e) {
        e.printStackTrace();
      }
    }
  }

  private Runnable run = new Runnable() {
    public void run() {
      int _pec=(int) (downloadLength*100 / fileLength);
      builder.setContentText("下载中……"+_pec+"%");
      builder.setProgress(100, _pec, false);//显示进度条,参数分别是最大值、当前值、是否显示具体进度(false显示具体进度,true就只显示一个滚动色带)
      manager.notify(_notificationID,builder.build());
      handler.postDelayed(run, 1000);
    }
  };

  @Override
  public void onDestroy() {
    Log.d(TAG, "onDestroy");
    handler.removeCallbacks(run);
    super.onDestroy();
  }

  public void initNotification(){
    builder = new NotificationCompat.Builder(this);
    builder.setSmallIcon(R.mipmap.ic_launcher).setContentTitle("下载文件").setContentText("下载中……");//图标、标题、内容这三个设置是必须要有的。
    manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
  }
}

相关github项目地址:serviceDownloaderDemo

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • [译]技术公司十年经验的职场生涯回顾
    本文是一位在技术公司工作十年的职场人士对自己职业生涯的总结回顾。她的职业规划与众不同,令人深思又有趣。其中涉及到的内容有机器学习、创新创业以及引用了女性主义者在TED演讲中的部分讲义。文章表达了对职业生涯的愿望和希望,认为人类有能力不断改善自己。 ... [详细]
  • Google Play推出全新的应用内评价API,帮助开发者获取更多优质用户反馈。用户每天在Google Play上发表数百万条评论,这有助于开发者了解用户喜好和改进需求。开发者可以选择在适当的时间请求用户撰写评论,以获得全面而有用的反馈。全新应用内评价功能让用户无需返回应用详情页面即可发表评论,提升用户体验。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • 本文介绍了设计师伊振华受邀参与沈阳市智慧城市运行管理中心项目的整体设计,并以数字赋能和创新驱动高质量发展的理念,建设了集成、智慧、高效的一体化城市综合管理平台,促进了城市的数字化转型。该中心被称为当代城市的智能心脏,为沈阳市的智慧城市建设做出了重要贡献。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • 关于我们EMQ是一家全球领先的开源物联网基础设施软件供应商,服务新产业周期的IoT&5G、边缘计算与云计算市场,交付全球领先的开源物联网消息服务器和流处理数据 ... [详细]
  • ZSI.generate.Wsdl2PythonError: unsupported local simpleType restriction ... [详细]
  • 推荐系统遇上深度学习(十七)详解推荐系统中的常用评测指标
    原创:石晓文小小挖掘机2018-06-18笔者是一个痴迷于挖掘数据中的价值的学习人,希望在平日的工作学习中,挖掘数据的价值, ... [详细]
  • 解决Cydia数据库错误:could not open file /var/lib/dpkg/status 的方法
    本文介绍了解决iOS系统中Cydia数据库错误的方法。通过使用苹果电脑上的Impactor工具和NewTerm软件,以及ifunbox工具和终端命令,可以解决该问题。具体步骤包括下载所需工具、连接手机到电脑、安装NewTerm、下载ifunbox并注册Dropbox账号、下载并解压lib.zip文件、将lib文件夹拖入Books文件夹中,并将lib文件夹拷贝到/var/目录下。以上方法适用于已经越狱且出现Cydia数据库错误的iPhone手机。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • 本文介绍了Linux系统中正则表达式的基础知识,包括正则表达式的简介、字符分类、普通字符和元字符的区别,以及在学习过程中需要注意的事项。同时提醒读者要注意正则表达式与通配符的区别,并给出了使用正则表达式时的一些建议。本文适合初学者了解Linux系统中的正则表达式,并提供了学习的参考资料。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
author-avatar
飞隔热条二
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有