文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

怎么在Android中实现一个多线程下载功能

2023-05-30 17:30

关注

怎么在Android中实现一个多线程下载功能?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。

布局

        界面上自上而下放置一个TextView,用来提示文本框中输入的信息,一个文本框用来输入网络中下载文件的路径,一个Button按钮,点击下载文件,一个ProgressBar显示下载进度,一个TextView显示下载的百分比。具体布局内容如下:

<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:paddingBottom="@dimen/activity_vertical_margin"  android:paddingLeft="@dimen/activity_horizontal_margin"  android:paddingRight="@dimen/activity_horizontal_margin"  android:paddingTop="@dimen/activity_vertical_margin"  android:orientation="vertical"  tools:context=".MainActivity" >   <TextView  android:layout_width="match_parent"  android:layout_height="wrap_content"  android:text="下载路径" />   <EditText  android:id="@+id/ed_path"  android:layout_width="match_parent"  android:layout_height="wrap_content"  android:text="http://192.168.0.170:8080/web/youdao.exe"/>  <Button  android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:text="下载"  android:onClick="download"/>   <ProgressBar  android:id="@+id/pb"  android:layout_width="match_parent"  android:layout_height="wrap_content"  />   <TextView  android:id="@+id/tv_info"  android:layout_width="match_parent"  android:layout_height="wrap_content"  android:gravity="center"  android:text="下载:0%"/>  </LinearLayout>

自定义ProgressBarListener监听器接口

         新建自定义ProgressBarListener监听器接口,这个接口中定义两个方法,void getMax(int length)用来获取下载文件的长度,void getDownload(int length);用来获取每次下载的长度,这个方法中主要是在多线程中调用,子线程中获取到的数据传递到这两个接口方法中,然后在这两个接口方法中通过Handler将相应的长度信息传递到主线程,更新界面显示信息,具体代码实现如下:

package com.example.inter;   public interface ProgressBarListener {    void getMax(int length);    void getDownload(int length); }

自定义线程类DownloadThread

          这里通过继承Thread的方式来实现自定义线程操作,在这个类中主要是实现文件的下载操作,在这个类中,定义了一系列与下载有关的实例变量来控制下载的数据,同时通过自定义监听器ProgressBarListener中的void getDownload(int length)方法来跟新界面显示的进度信息。
具体实现如下:

package com.example.download;  import java.io.File; import java.io.FileOutputStream; import java.io.InputStream; import java.io.RandomAccessFile; import java.net.HttpURLConnection; import java.net.URL;  import com.example.inter.ProgressBarListener;   public class DownloadThread extends Thread {  //下载的线程id  private int threadId;  //下载的文件路径  private String path;  //保存的文件  private File file;  //下载的进度条更新的监听器  private ProgressBarListener listener;  //每条线程下载的数据量  private int block;  //下载的开始位置  private int startPosition;  //下载的结束位置  private int endPosition;   public DownloadThread(int threadId, String path, File file, ProgressBarListener listener, int block) {  this.threadId = threadId;  this.path = path;  this.file = file;  this.listener = listener;  this.block = block;    this.startPosition = threadId * block;  this.endPosition = (threadId + 1) * block - 1;  }   @Override  public void run() {  super.run();  try {   //创建RandomAccessFile对象   RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");   //跳转到开始位置   accessFile.seek(startPosition);   URL url = new URL(path);   //打开http链接   HttpURLConnection conn = (HttpURLConnection) url.openConnection();   //设置超时时间   conn.setConnectTimeout(5000);   //指定请求方式为GET方式   conn.setRequestMethod("GET");   //指定下载的位置   conn.setRequestProperty("Range", "bytes="+startPosition + "-" + endPosition);   //不用再去判断状态码是否为200   InputStream in = conn.getInputStream();   byte[] buffer = new byte[1024];   int len = 0;   while((len = in.read(buffer)) != -1){   accessFile.write(buffer, 0, len);   //更新下载进度   listener.getDownload(len);   }   accessFile.close();   in.close();  } catch (Exception e) {   // TODO: handle exception   e.printStackTrace();  }  } }

新建DownloadManager类

         这个类主要是对下载过程的管理,包括下载设置下载后文件要保存的位置,计算多线程中每个线程的数据下载量等等。
具体实现如下:

package com.example.download;  import java.io.File; import java.io.RandomAccessFile; import java.net.HttpURLConnection; import java.net.URL;  import android.os.Environment;  import com.example.inter.ProgressBarListener;   public class DownloadManager {  //下载线程的数量  private static final int TREAD_SIZE = 3;  private File file;    public void download(String path, ProgressBarListener listener) throws Exception{  URL url = new URL(path);  HttpURLConnection conn = (HttpURLConnection) url.openConnection();  conn.setConnectTimeout(5000);  conn.setRequestMethod("GET");  if(conn.getResponseCode() == 200){   int filesize = conn.getContentLength();   //设置进度条的最大长度   listener.getMax(filesize);   //创建一个和服务器大小一样的文件   file = new File(Environment.getExternalStorageDirectory(), this.getFileName(path));   RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");   accessFile.setLength(filesize);   //要关闭RandomAccessFile对象   accessFile.close();     //计算出每条线程下载的数据量   int block = filesize % TREAD_SIZE == 0 ? (filesize / TREAD_SIZE) : (filesize / TREAD_SIZE +1 );     //开启线程下载   for(int i = 0; i < TREAD_SIZE; i++){   new DownloadThread(i, path, file, listener, block).start();   }  }  }     private String getFileName(String path){  return path.substring(path.lastIndexOf("/") + 1);  } }

完善MainActivity

      在这个类中首先,找到页面中的各个控件,实现Button按钮的onClick事件,在onClick事件中开启一个线程进行下载操作,同时子线程中获取到的数据,通过handler与Message机制传递到主线程,更新界面显示。
具体实现如下:

package com.example.multi;  import android.app.Activity; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.view.Menu; import android.view.View; import android.widget.EditText; import android.widget.ProgressBar; import android.widget.TextView; import android.widget.Toast;  import com.example.download.DownloadManager; import com.example.inter.ProgressBarListener;   public class MainActivity extends Activity {   protected static final int ERROR_DOWNLOAD = 0;  protected static final int SET_PROGRESS_MAX = 1;  protected static final int UPDATE_PROGRESS = 2;   private EditText ed_path;  private ProgressBar pb;  private TextView tv_info;  private DownloadManager manager;  //handler操作  private Handler mHandler = new Handler(){    public void handleMessage(android.os.Message msg) {   switch (msg.what) {   case ERROR_DOWNLOAD:   //提示用户下载失败   Toast.makeText(MainActivity.this, "下载失败", Toast.LENGTH_SHORT).show();   break;   case SET_PROGRESS_MAX:   //得到最大值   int max = (Integer) msg.obj;   //设置进度条的最大值   pb.setMax(max);   break;   case UPDATE_PROGRESS:   //获取当前下载的长度   int currentprogress = pb.getProgress();   //获取新下载的长度   int len = (Integer) msg.obj;   //计算当前总下载长度   int crrrentTotalProgress = currentprogress + len;   pb.setProgress(crrrentTotalProgress);      //获取总大小   int maxProgress = pb.getMax();   //计算百分比   float value = (float)currentprogress / (float)maxProgress;   int percent = (int) (value * 100);   //显示下载的百分比   tv_info.setText("下载:"+percent+"%");   break;   default:   break;   }  };  };  @Override  protected void onCreate(Bundle savedInstanceState) {  super.onCreate(savedInstanceState);  setContentView(R.layout.activity_main);  this.ed_path = (EditText) super.findViewById(R.id.ed_path);  this.pb = (ProgressBar) super.findViewById(R.id.pb);  this.tv_info = (TextView) super.findViewById(R.id.tv_info);  this.manager = new DownloadManager();    }   @Override  public boolean onCreateOptionsMenu(Menu menu) {  // Inflate the menu; this adds items to the action bar if it is present.  getMenuInflater().inflate(R.menu.main, menu);  return true;  }   public void download(View v){  final String path = ed_path.getText().toString();  //下载  new Thread(new Runnable() {   @Override   public void run() {   // TODO Auto-generated method stub   try {    manager.download(path, new ProgressBarListener() {    @Override    public void getMax(int length) {     // TODO Auto-generated method stub     Message message = new Message();     message.what = SET_PROGRESS_MAX;     message.obj = length;     mHandler.sendMessage(message);    }        @Override    public void getDownload(int length) {     // TODO Auto-generated method stub     Message message = new Message();     message.what = UPDATE_PROGRESS;     message.obj = length;     mHandler.sendMessage(message);    }    });   } catch (Exception e) {    // TODO: handle exception    e.printStackTrace();    Message message = new Message();    message.what = ERROR_DOWNLOAD;    mHandler.sendMessage(message);   }   }  }).start();  } }

增加权限

最后,别忘了给应用授权,这里要用到Android联网授权和向SD卡中写入文件的权限。
具体实现如下:

<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android"  package="com.example.multi"  android:versionCode="1"  android:versionName="1.0" >   <uses-sdk  android:minSdkVersion="8"  android:targetSdkVersion="18" />  <uses-permission android:name="android.permission.INTERNET"/>  <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>  <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>  <application  android:allowBackup="true"  android:icon="@drawable/ic_launcher"  android:label="@string/app_name"  android:theme="@style/AppTheme" >  <activity   android:name="com.example.multi.MainActivity"   android:label="@string/app_name" >   <intent-filter>   <action android:name="android.intent.action.MAIN" />    <category android:name="android.intent.category.LAUNCHER" />   </intent-filter>  </activity>  </application>  </manifest>

四、运行效果

怎么在Android中实现一个多线程下载功能

怎么在Android中实现一个多线程下载功能

怎么在Android中实现一个多线程下载功能

关于怎么在Android中实现一个多线程下载功能问题的解答就分享到这里了,希望以上内容可以对大家有一定的帮助,如果你还有很多疑惑没有解开,可以关注编程网行业资讯频道了解更多相关知识。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯