Android使用Handler实现下载文件功能

 更新时间:2018年06月10日 15:12:01   作者:轻扰时光  
这篇文章主要为大家详细介绍了Android使用Handler实现下载文件功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android实现下载文件的具体代码,供大家参考,具体内容如下

1.实现效果

直接上图:

2.代码实现

在AndroidManifest.xml文件中声明申请的权限,如下所示:

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

新建一个名为DownloadDemo的项目,activity_main.xml代码如下:

<?xml version="1.0" encoding="utf-8"?>
<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:gravity="center_horizontal"
 android:orientation="vertical"
 tools:context=".MainActivity">

 <Button
  android:id="@+id/button"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:layout_marginTop="10dp"
  android:text="下载" />

 <ProgressBar
  android:id="@+id/progressBar"
  style="?android:attr/progressBarStyleHorizontal"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:layout_margin="10dp"
  android:max="100"
  android:progress="0" />
</LinearLayout>

MainActivity.class代码如下:

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

 public static final String APP_URL = "http://download.sj.qq.com/upload/connAssitantDownload/upload/MobileAssistant_1.apk";
 public static final int DOWNLOAD_MESSAGE_CODE = 100001;
 private static final int DOWNLOAD_MESSAGE_FAIL_CODE = 100002;
 public static final int REQUEST_CODE = 1;
 private Button button;
 private ProgressBar progressBar;
 private MyHandler mHandler;

 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  initView();
  askPermission();
  mHandler = new MyHandler(this);
 }

 public static class MyHandler extends Handler {
  final WeakReference<MainActivity> mWeakReference;

  public MyHandler(MainActivity activity) {
   this.mWeakReference = new WeakReference<>(activity);
  }

  @Override
  public void handleMessage(Message msg) {
   MainActivity activity = mWeakReference.get();
   super.handleMessage(msg);
   switch (msg.what) {
    case DOWNLOAD_MESSAGE_CODE:
     activity.progressBar.setProgress((Integer) msg.obj);
     if (((Integer) msg.obj) == 100) {
      Toast.makeText(activity, "文件下载已完成!", Toast.LENGTH_SHORT).show();
     }
     break;
    case DOWNLOAD_MESSAGE_FAIL_CODE:
     Toast.makeText(activity, "文件下载失败!", Toast.LENGTH_SHORT).show();
     break;
   }
  }
 }

 /**
  * 申请权限
  */
 private void askPermission() {
  //将所需申请的权限添加到List集合中
  List<String> permissionList = new ArrayList<>();
  if (ContextCompat.checkSelfPermission(MainActivity.this,
    Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
   permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
  }
  //判断权限列表是否为空,若不为空,则向用户申请权限,否则则直接执行操作
  if (!permissionList.isEmpty()) {
   String[] permissions = permissionList.toArray(new String[permissionList.size()]);
   ActivityCompat.requestPermissions(MainActivity.this, permissions, REQUEST_CODE);
  } else {
   //TODO
   button.setOnClickListener(this);
  }
 }

 /**
  * 用户对请求权限的dialog做出响应之后,会处理请求权限的响应
  *
  * @param requestCode 请求码
  * @param permissions 权限的集合
  * @param grantResults 权限授予的结果
  */
 @Override
 public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  switch (requestCode) {
   case REQUEST_CODE:
    if (grantResults.length > 0) {
     for (int result : grantResults) {
      if (result != PackageManager.PERMISSION_GRANTED) {
       Toast.makeText(this, "必须同意所有权限才能使用本程序", Toast.LENGTH_SHORT).show();
       finish();
       return;
      }
     }
     //TODO
     button.setOnClickListener(this);
    } else {
     Toast.makeText(this, "权限申请失败!", Toast.LENGTH_SHORT).show();
     finish();
    }
    break;
   default:
    break;
  }
 }

 /**
  * 点击事件
  *
  * @param v
  */
 @Override
 public void onClick(View v) {
  switch (v.getId()) {
   case R.id.button:
    new Thread(new Runnable() {
     @Override
     public void run() {
      download(APP_URL);
     }
    }).start();
    break;
  }
 }

 /**
  * 下载
  *
  * @param appUrl
  */
 private void download(String appUrl) {
  try {
   //实例化一个url对象
   URL url = new URL(appUrl);
   //获取URLConnection对象
   URLConnection urlConnection = url.openConnection();
   //获取输入流
   InputStream inputStream = urlConnection.getInputStream();
   //获取文件的总长度
   int contentLength = urlConnection.getContentLength();
   //设置软件下载到手机存储的位置文件夹名称
   String downloadFolderName = Environment.getExternalStorageDirectory()
     + File.separator + "MyApp" + File.separator;
   File file = new File(downloadFolderName);
   //若文件夹不存在则进行创建
   if (!file.exists()) {
    file.mkdir();
   }
   String fileName = downloadFolderName + "chaixingsi.apk";
   File apkFile = new File(fileName);
   if (apkFile.exists()) {
    apkFile.delete();
   }
   int downloadContentLength = 0;
   int length = 0;
   byte[] bytes = new byte[1024];
   OutputStream outputStream = new FileOutputStream(fileName);
   while ((length = inputStream.read(bytes)) != -1) {
    outputStream.write(bytes, 0, length);
    downloadContentLength += length;
    /**
     * 发送消息通知主线程去更新UI
     */
    Message message = Message.obtain();
    message.obj = downloadContentLength * 100 / contentLength;
    message.what = DOWNLOAD_MESSAGE_CODE;
    mHandler.sendMessage(message);
   }
   inputStream.close();
   outputStream.close();
  } catch (MalformedURLException e) {
   //下载失败的话也发送消息
   notifyDownloadFailed();
   e.printStackTrace();
  } catch (IOException e) {
   notifyDownloadFailed();
   e.printStackTrace();
  }
 }

 /**
  * 通知下载失败
  */
 private void notifyDownloadFailed() {
  Message message = Message.obtain();
  message.what = DOWNLOAD_MESSAGE_FAIL_CODE;
  mHandler.sendMessage(message);
 }

 /**
  * 初始化视图
  */
 private void initView() {
  button = findViewById(R.id.button);
  progressBar = findViewById(R.id.progressBar);
 }
}

3.快捷键总结

提取方法:Ctrl+Alt+M;
提取变量:Ctrl+Alt+V;
提取常量:Ctrl+Alt+C;

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

相关文章

  • Android开发VR实战之播放360度全景视频

    Android开发VR实战之播放360度全景视频

    这篇文章主要为大家详细介绍了Android开发VR实战之播放360度全景视频,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • Android WebView 上传文件支持全解析

    Android WebView 上传文件支持全解析

    这篇文章主要针对Android WebView 上传文件支持进行全面解析,感兴趣的小伙伴们可以参考一下
    2016-06-06
  • Android通知栏增加快捷开关的功能实现教程

    Android通知栏增加快捷开关的功能实现教程

    对于Android来说其中一项很方便的操作便是下拉菜单,下拉菜单栏可以快捷打开某项设置,这篇文章主要给大家介绍了关于Android通知栏增加快捷开关的功能实现,需要的朋友可以参考下
    2023-01-01
  • Kotlin开发笔记之委托属性与区间(译)

    Kotlin开发笔记之委托属性与区间(译)

    最近在学习kotlin,发现了一些比较重要的知识点,所以下面这篇文章主要给大家介绍了关于Kotlin开发笔记之委托属性与区间的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-12-12
  • Android自定义View播放Gif动画的示例

    Android自定义View播放Gif动画的示例

    本篇文章主要介绍了Android自定义View播放Gif动画的示例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • Intent传递对象之Serializable和Parcelable的区别

    Intent传递对象之Serializable和Parcelable的区别

    Intent在不同的组件中传递对象数据的应用非常普遍,大家都知道在intent传递对象的方法有两种:1、实现Serializable接口、2、实现Parcelable接口,接下来通过本文给大家介绍Intent传递对象之Serializable和Parcelable的区别,感兴趣的朋友一起学习吧
    2016-01-01
  • Android AndBase框架使用封装好的函数完成Http请求(三)

    Android AndBase框架使用封装好的函数完成Http请求(三)

    这篇文章主要介绍了Android AndBase框架使用封装好的函数完成Http请求的相关资料,感兴趣的小伙伴们可以参考一下
    2016-03-03
  • Android实现图片双指缩放

    Android实现图片双指缩放

    这篇文章主要为大家详细介绍了Android实现图片双指缩放,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • Android编程基于自定义View实现绚丽的圆形进度条功能示例

    Android编程基于自定义View实现绚丽的圆形进度条功能示例

    这篇文章主要介绍了Android编程基于自定义View实现绚丽的圆形进度条功能,结合实例形式详细分析了Android自定义view实现圆形进度条的具体步骤与相关操作技巧,需要的朋友可以参考下
    2017-01-01
  • Android性能之冷启动优化详析

    Android性能之冷启动优化详析

    这篇文章主要给大家介绍了关于Android性能之冷启动优化的相关资料,文中通过示例代码介绍的非常详细,对各位Android开发者们具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-11-11

最新评论