Android类FileDownloadList分析
作者:hebedich 发布时间:2022-06-17 02:11:36
先上代码,再来分析
public class FileDownloadList {
/**上下文*/
private Context mContext;
/**请求对象*/
private BaseRequestLims fileRequest = null;
/**进度条对话框*/
private AlertDialog progressDialog = null;
/**进度条控件变量*/
private ProgressBar mProgress;
/**百分比显示控件*/
private TextView mProgressPercent;
private File localFile = null;
/**接收HttpHelper中获取到文件大小后发送的广播,确定文件大小*/
private DownLoadReceiver receiver;
/**文件大小*/
private long fileLength = -1L;
/**是否已注册广播标志*/
private boolean castFlag = false;
/**是否显示进度条标志*/
private boolean showDialog = false;
/**文件下载完的回调接口*/
private Runnable mCallback = null;
private Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
int tempSize = (int)localFile.length();
if(tempSize < fileLength){
//文件下载中
if(showDialog){
//显示了进度条的情况下,更新进度条
int progress = (int)((Double.valueOf(tempSize) / Double.valueOf(fileLength)) * 100);
mProgress.setProgress(tempSize);
mProgressPercent.setText(progress + "%");
}
}else{
//下载文件完毕
if(castFlag){//如已注册广播,注销广播
mContext.unregisterReceiver(receiver);
castFlag = false;
}
if(showDialog){
mProgress.setProgress((int)fileLength);
mProgressPercent.setText("100%");
progressDialog.dismiss();
}
if(mCallback != null){
try{
Thread.sleep(500);
mCallback.run();
}catch (Exception e) {
e.printStackTrace();
}
}
}
}
};
/**
* 构造器
* @param activity
*/
/**
* 构造器
* @param activity
* @param showDialog 显示进度条标志
*/
public FileDownloadList(Context context, boolean showDialog){
mContext = context;
this.showDialog = showDialog;
fileRequest = new BaseRequestLims(context,ClientServiceType.FILE_DOWN);
fileRequest.setMethodType(BaseRequestLims.METHOD_TYPE_POST);
fileRequest.setContext(mContext);
}
public BaseRequestLims getFileRequest(){
return fileRequest;
}
/**
* 通过关联类型来下载文件
* @param fileName 文件名称或文件在服务器上的相对路径加名称
* @param saveDir 保存在本地的文件目录
* @param saveName 保存在本地的文件名称
* @param gllx 关联类型
* @param callback 下载后的处理线程
*/
public void downloadFile(String fileName, String saveDir, String saveName, Runnable callback){
if(callback != null){
mCallback = callback;
}
File saveDirFile = new File(saveDir);
//judge the save dir path exist or not
if(!saveDirFile.exists()){
saveDirFile.mkdirs();
}
localFile = new File(saveDir,saveName);
if(localFile.isDirectory()){
new AlertDialog.Builder(mContext).setTitle("提示").setMessage("the save file is directory").show();
return;
}
if(fileRequest.getServiceType()==null){
fileRequest.setServiceType(ClientServiceType.FILE_DOWN);
}
fileRequest.addParameter("fpath", fileName);
fileRequest.addParameter("fname", saveName);
fileRequest.setStreamPath(localFile.getAbsolutePath());
fileRequest.setStream(true);
if(localFile.exists()){
if(localFile.length() == 0){
invokeFile(fileRequest);
}else{
//文件存在直接打开
if(showDialog)
buildProgressDialog().show();
mHandler.sendMessage(mHandler.obtainMessage());
}
}else{
invokeFile(fileRequest);
}
}
/**
* 进入文件下载子线程
* @param request
*/
private void invokeFile(final BaseRequestLims request){
try{
if(showDialog){
buildProgressDialog().show();
}
receiver = new DownLoadReceiver();
IntentFilter filter = new IntentFilter();
filter.addAction("SAVE_DOWNLOAD_FILE");
mContext.registerReceiver(receiver, filter);
castFlag = true;
//下载的子线程
new Thread(){
@Override
public void run() {
super.run();
HttpHelper.invoke(request);
}
}.start();
}catch (Exception e) {
e.printStackTrace();
}
}
/**
* 创建进度对话框
* @return
*/
private AlertDialog buildProgressDialog(){
AlertDialog.Builder builder = new Builder(mContext);
builder.setTitle("正在下载文件,请稍候...");
RelativeLayout container = new RelativeLayout(mContext);
mProgress = new ProgressBar(mContext);
mProgress.setId("progress".hashCode());
BeanUtils.setFieldValue(mProgress, "mOnlyIndeterminate", Boolean.valueOf(false));
mProgress.setIndeterminate(false);
LayerDrawable layerDrawable = (LayerDrawable)mContext.getResources().getDrawable(android.R.drawable.progress_horizontal);
ClipDrawable clipDrawable = (ClipDrawable)layerDrawable.getDrawable(2);
clipDrawable.setColorFilter(Color.parseColor("#32B5E5"), Mode.SRC_IN);
mProgress.setProgressDrawable(layerDrawable);
mProgress.setPadding(0, 0, 0, 0);
mProgress.setIndeterminateDrawable(
mContext.getResources().getDrawable(android.R.drawable.progress_indeterminate_horizontal));
mProgressPercent = new TextView(mContext);
mProgressPercent.setId("percent".hashCode());
mProgressPercent.setText("0%");
mProgressPercent.setTextSize(18);
int containerPadding = DimensionUtils.dip2Px(mContext, 10);
container.setPadding(containerPadding, containerPadding, containerPadding, containerPadding);
LayoutParams progressLayoutParams = new LayoutParams(
LayoutParams.MATCH_PARENT, DimensionUtils.dip2Px(mContext, 4));
progressLayoutParams.addRule(RelativeLayout.CENTER_VERTICAL);
progressLayoutParams.addRule(RelativeLayout.LEFT_OF, mProgressPercent.getId());
mProgress.setLayoutParams(progressLayoutParams);
LayoutParams percentLayoutParams = new LayoutParams(
LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
percentLayoutParams.addRule(RelativeLayout.ALIGN_PARENT_RIGHT);
percentLayoutParams.addRule(RelativeLayout.CENTER_VERTICAL);
mProgressPercent.setLayoutParams(percentLayoutParams);
container.addView(mProgressPercent);
container.addView(mProgress);
builder.setView(container);
builder.setNegativeButton("取消", new OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
dialog.dismiss();
}
});
progressDialog = builder.create();
return progressDialog;
}
class DownLoadReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
//显示进度条
fileLength = intent.getLongExtra("FILE_LENGTH", -1);
if(showDialog){
mProgress.setMax((int)fileLength);
}
//更新进度条的线程
new Thread(){
@Override
public void run() {
super.run();
while(true){
try{
Thread.sleep(500);
}catch (Exception e) {
e.printStackTrace();
}
mHandler.sendMessage(mHandler.obtainMessage());
//获取下载文件的大小
int loadedSize = (int)localFile.length();
if(loadedSize >= fileLength){
break;
}
}
}
}.start();
}
}
public DownLoadReceiver getReciver()
{
return receiver;
}
}
它的逻辑:
创建一个FileDownloadList对象后,就可以直接使用该下述方法来实现下载功能。
downloadFile(String fileName, String saveDir, String saveName, Runnable callback)
在实现上是这么个意思:
1.在当前上下文,开启下载线程。当获取到要下载的文件的大小时,发送一个广播过来(这部分没有展示在上述代码中)。
2.在当前上下文中,注册一个广播 * ,监听广播标识为SAVE_DOWNLOAD_FILE的广播。首次监听到发出来的广播后,首次发送过来的广播,包含了要下载的文件的大小信息,然后就每隔5毫秒检测本地文件的大小,直到本地文件的大小(loadedSize)大于等于要下载的文件(fileLength)大小时,退出该循环。
在不断检测的过程中,通过mHandler.sendMessage(mHandler.obtainMessage()); ,让UI线程更新进度条。
下载线程,会不断将服务器返回的数据流,写到本地文件中,所以,本地文件的大小会不断变化,直到,它的大小跟要下载的文件的大小相等时,就退出这个不断检测本地文件大小的线程。
其它没有在上述代码中表现出来的内容(在其它部分的代码中):
1.在invokeFile( final BaseRequestLims request)方法中,开了一个如下的下载线程.该下载线程,会将服务器返回的文件流,写到本地文件(localFile)中;然后,它还会发送一个标识为SAVE_DOWNLOAD广播,包含的信息有要下载文件的文件大小fileLength。
//下载的子线程
new Thread(){
@Override
public void run() {
super.run();
HttpHelper.invoke(request);
}
}.start();
上述代码存在的问题:
1.上下文,使用的是某个Activity,如果发生系统调用了该Activity的onDestroy()时,下载线程还没有完成,也就意味着,loadedSize的大小还是小于fileLength。从而,那个不断检测本地文件大小的线程就一直在执行着。
即是检测本地文件大小的线程和下载线程还在执行着:
检测本地文件大小的线程:
new Thread(){
@Override
public void run() {
super.run();
while(true){
try{
Thread.sleep(500);
}catch (Exception e) {
e.printStackTrace();
}
mHandler.sendMessage(mHandler.obtainMessage());
//获取下载文件的大小
int loadedSize = (int)localFile.length();
if(loadedSize >= fileLength){
break;
}
}
}
}.start();
下载线程:
new Thread(){
@Override
public void run() {
super.run();
HttpHelper.invoke(request);
}
}.start();
那么,会出现什么问题呢?
1).我可以确定的就是,mContext会出现泄漏。
2). DownLoadReceiver不能正常被取消注册。
分析,待续。


猜你喜欢
- 前言在移动互联网浪潮中,联网APP已经把单机拍死在沙滩上,很多公司都希望自家应用能够有一套帐号系统,可是许多用户却并不一定买账:
- 前段时间做了一个练手的小项目,名叫Book_Bar,用来卖书的,采用的是三层架构,也就是Models,IDAL,DAL,BLL 和 Web
- cookie机制和session机制的区别具体来说cookie机制采用的是在客户端保持状态的方案,而session机制采用的是在服务器端保持
- 简介我们知道Java中Collection接口下的很多集合都是线程不安全的, 比如 java.util.ArrayList不是线程安全的,
- dynamic是FrameWork4.0的新特性。dynamic的出现让C#具有了弱语言类型的特性。编译器在编译的时候不再对类型进行检查,编
- 前言大家好,我是小郭,前面我们学习了利用Semaphore来防止多线程同时操作一个资源,通常我们都会利用并行来优化性能,但是对于串行化的业务
- 前言:需求是这样的,在与第三方对接过程中,对方提供了token进行时效性验证,过一段时间token就会失效.后台有定时任务在获取,但是偶尔会
- 如今代码圈很多做网络爬虫的例子,今天小编给大家分享的是如何用C#做网络爬虫。注意这次的分享只是分享思路,并不是一整个例子,因为如果要讲解一整
- JVM默认物理内存JVM初始分配的内存由-Xms指定,默认是物理内存的1/64;JVM最大分配的内存由-Xmx指定,默认是物理内存的1/4。
- Android ListView的Item点击效果的定制
- 目录无SpringMVC全局异常时的流程图SpringMVC全局异常流程图其实是一个ModelAndView对象配置文件applicatio
- 在android开发中,当不满足触发条件就按返回键的时候,就要对此进行检测。尤其是当前Activity需要往前一个Activity传送消息时
- Android中手机震动的设置(Vibrator)的步骤: a、通过系统服务获得手机震动服务,Vibrator vibrator = (Vi
- 断言的概念断言用于证明和测试程序的假设,比如“这里的值大于 5”。断言可以在运行时从代码中完全删除,所以对代码的运行速度没有影响。断言的使用
- NullPointerException是当您尝试使用指向内存中空位置的引用(null)时发生的异常,就好像它引用了一个对象一样。当我们声明
- 背景为了了解Seata AT模式的原理,我通过源码解读的方式画出了Seata AT模式启动的图示:如果是基于Springboot项目的话,项
- 安卓提供的列表选择框(Spinner)相当于web端用户注册时的选择下拉框,比如注册候选择省份城市等。如下图便是一个列表选择框下拉列表的列表
- —学习并使用mybatis-plus的一些高级功能的用法例如: AR模式、 乐观锁 、逻辑删除 、自动填充、数据保护等功能为了方便演示,咱们
- 群主发普通红包,某群有多名成员,群主给成员发普通红包,普通红包的规则:群主的一笔金额,从群主余额中扣除,平均分成n等份,让成员领取;成员领取
- 本文实例为大家分享了java web实现简单留言板的具体代码,供大家参考,具体内容如下一、目标用户可以登录并记住密码进入留言板,添加留言,点