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

所属分类: 软件编程 / Android 阅读数: 72
收藏 0 赞 0 分享

本文实例为大家分享了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网络编程之获取网络上的Json数据实例

这篇文章主要介绍了Android网络编程之获取网络上的Json数据实例,本文用完整的代码实例讲解了在Android中读取网络中Json数据的方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中的windowSoftInputMode属性详解

这篇文章主要介绍了Android中的windowSoftInputMode属性详解,本文对windowSoftInputMode的9个属性做了详细总结,需要的朋友可以参考下
收藏 0 赞 0 分享

Android网络编程之UDP通信模型实例

这篇文章主要介绍了Android网络编程之UDP通信模型实例,本文给出了服务端代码和客户端代码,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中使用ListView实现漂亮的表格效果

这篇文章主要介绍了Android中使用ListView实现漂亮的表格效果,本文用详细的代码实例创建了一个股票行情表格,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中刷新界面的二种方法

这篇文章主要介绍了Android中刷新界面的二种方法,本文使用Handler、postInvalidate两种方法实现界面刷新,需要的朋友可以参考下
收藏 0 赞 0 分享

Android SDK三种更新失败及其解决方法

这篇文章主要介绍了Android SDK三种更新失败及其解决方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(一)

Android3.0(API level 11)开始,Android设备不再需要专门的菜单键。随着这种变化,Android app应该取消对传统6项菜单的依赖。取而代之的是提供anction bar来提供基本的用户功能
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(二)

这次将继续上一篇文章没有讲完的Menu的学习,上下文菜单(Context menu)和弹出菜单(Popup menu)
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(三)

今天继续昨天没有讲完的Menu的学习,主要是Popup Menu的学习,需要的朋友可以参考下
收藏 0 赞 0 分享

Android显示网络图片实例

这篇文章主要介绍了Android显示网络图片的方法,以实例形式展示了Android程序显示网络图片的方法,非常具有实用价值,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多