Android使用Handler实现下载文件功能
作者:轻扰时光 发布时间:2023-05-21 03:23:08
标签: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;
来源:https://blog.csdn.net/qingjianduoyun/article/details/80628571
0
投稿
猜你喜欢
- 一、输入映射parameterType指定输入参数的Java类型,可以使用别名或者类的全限定名。它可以接收简单类型、POJO、H
- JVM的对dll文件的装载过程操作系统装入JVM是通过jdk中Java.exe来完成,通过下面4步来完成JVM环境.1.创建JVM装载环境和
- 前言需求是上传Excel文件并读取Excel文件中的内容,根据获取的数据执行完某些业务操作后再将一些数据写回到excel中。前台使用Form
- 前言上一篇文章用贝塞尔曲线画了一个看起来不错的小红点功能,技术上没什么难度,主要就是数学上的计算。这篇文章也差不多,模仿了一个常用的滑动解锁
- 开放端口安全组没开放端口是原罪!!!导致好多BUG费时费力。Hbase悄悄 * 的用了好多端口,比如被我抓到的42239,直接搜索报错药不对症
- 简介在上一篇文章中,我们列举了flutter中的所有layout类,并且详细介绍了两个非常常用的layout:Row和Column。掌握了上
- Android MTU 值修改的实例详解通信术语 最大传输单元(Maximum Transmission Unit,MTU)是指一种通信协议
- 最近在研究dubbo时,发现了JAVA的SPI特性。SPI的全名为Service Provider Interface,是JDK内置的一种服
- 本文实例讲述了Android使用WebView播放flash及判断是否安装flash插件的方法。分享给大家供大家参考。具体实现方法如下:一、
- 整合Spring Data JPAJPA (Java Persistence API)和 Spring Data 是两个范畴的概念。Hibe
- 概述:Flutter中常用的滑动布局 ScrollView 有 SingleChildScrollView、NestedScrollView
- 实现了Java web开发账号单一登录的功能,防止同一账号重复登录,后面登录的踢掉前面登录的,使用过滤器Filter实现的。可以先下载项目下
- 目录源码使用源码@Target({ElementType.TYPE, ElementType.METHOD})@Retention(Rete
- 一、项目运行环境配置:Jdk1.8 + Tomcat8.5 + Mysql + HBuilderX(Webstorm也行)+ Eclispe
- java输入流报错:Exception in thread "main" java.util.NoSuchElement
- Android开发,触控无处不在。对于一些 不咋看源码的同学来说,多少对这块都会有一些疑惑。View事件的分发机制,不仅在做业务需求中会碰到
- 主要是因为GZipStream的构造函数中第一个需要传入一个Stream,第二个是指定操作方式:压缩还是解压缩。当时的疑问点主要有:1.我传
- 面向对象语言对事物的体现都是以对象的形式,所以为了方便对多个对象的操作,就对对象进行存储,集合就是存储对象最常用的一种方式。数组虽然也可以存
- 目录为什么要实现调用链跟踪?如何实现?第一步,看图、看场景,用户浏览器的一次请求行为所走的路径是什么样的第二步,实现。不想看代码可直接拉最后
- 官方 JSON.NET 地址 http://james.newtonking.com/pages/json-net.aspxXML TO J