Android开发仿下载助手多线程下载及原理

仿下载助手界面效果

开发仿下载助手多线程下载及原理-android 双开助手原理">

线程池 ThreadPoolExecutor

在下面介绍实现下载原理的时候,我想尝试倒着来说,这样是否好理解一点?

我们都知道,下载助手,比如360, 百度的 手机助手,下载APP 的时候 ,都可以同时下载多个,所以,下载肯定是多线程的,所以我们就需要一个线程工具类 来管理我们的线程,这个工具类的核心,就是 线程池。

线程池ThreadPoolExecutor ,先简单学习下这个线程池的使用

 /**
   * Parameters:
      corePoolSize  
         the number of threads to keep in the pool, even if they are idle, unless allowCoreThreadTimeOut is set
      maximumPoolSize  
          the maximum number of threads to allow in the pool
      keepAliveTime
          when the number of threads is greater than the core, this is the maximum time that excess idle threads will wait for new tasks before terminating.
      unit  
          the time unit for the keepAliveTime argument
      workQueue  
          the queue to use for holding tasks before they are executed. This queue will hold only the Runnable tasks submitted                   by the execute method.
      handler  
         the handler to use when execution is blocked because the thread bounds and queue capacities are reached
    Throws:
      IllegalArgumentException - if one of the following holds:
      corePoolSize < 0
      keepAliveTime < 0
      maximumPoolSize <= 0
      maximumPoolSize < corePoolSize
      NullPointerException - if workQueue or handler is null
   */  
  ThreadPoolExecutor threadpool=new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue, handler)  

上面是 ThreadPoolExecutor的参数介绍,

第一个参数 corePoolSize : 空闲时 存在的线程数目、
第二个参数 maximumPoolSize :允许同时存在的最大线程数、
第三个参数 keepAliveTime: 这个参数是 允许空闲线程存活的时间、
第四个参数 unit : 是 时间的单位 、
第五个参数 workQueue :这个是一个容器,它里面存放的是、 threadpool.execute(new Runnable()) 执行的线程.new Runnable()、
第六个参数 handler:当执行被阻塞时,该处理程序将被阻塞,因为线程的边界和队列容量达到了 。

工具类 ThreadManager

介绍完了 线程池参数,那我们就先创建一个线程管理的工具类 ThreadManager

public class ThreadManager {  
    public static final String DEFAULT_SINGLE_POOL_NAME = "DEFAULT_SINGLE_POOL_NAME";  
 
    private static ThreadPoolProxy mLongPool = null;  
    private static Object mLongLock = new Object();  
 
    private static ThreadPoolProxy mShortPool = null;  
    private static Object mShortLock = new Object();  
 
    private static ThreadPoolProxy mDownloadPool = null;  
    private static Object mDownloadLock = new Object();  
 
    private static Map<String, ThreadPoolProxy> mMap = new HashMap<String, ThreadPoolProxy>();  
    private static Object mSingleLock = new Object();  
 
    /** 获取下载线程 */  
    public static ThreadPoolProxy getDownloadPool() {  
        synchronized (mDownloadLock) {  
            if (mDownloadPool == null) {  
                mDownloadPool = new ThreadPoolProxy(3, 3, 5L);  
            }  
            return mDownloadPool;  
        }  
    }  
 
    /** 获取一个用于执行长耗时任务的线程池,避免和短耗时任务处在同一个队列而阻塞了重要的短耗时任务,通常用来联网操作 */  
    public static ThreadPoolProxy getLongPool() {  
        synchronized (mLongLock) {  
            if (mLongPool == null) {  
                mLongPool = new ThreadPoolProxy(5, 5, 5L);  
            }  
            return mLongPool;  
        }  
    }  
 
    /** 获取一个用于执行短耗时任务的线程池,避免因为和耗时长的任务处在同一个队列而长时间得不到执行,通常用来执行本地的IO/SQL */  
    public static ThreadPoolProxy getShortPool() {  
        synchronized (mShortLock) {  
            if (mShortPool == null) {  
                mShortPool = new ThreadPoolProxy(2, 2, 5L);  
            }  
            return mShortPool;  
        }  
    }  
 
    /** 获取一个单线程池,所有任务将会被按照加入的顺序执行,免除了同步开销的问题 */  
    public static ThreadPoolProxy getSinglePool() {  
        return getSinglePool(DEFAULT_SINGLE_POOL_NAME);  
    }  
 
    /** 获取一个单线程池,所有任务将会被按照加入的顺序执行,免除了同步开销的问题 */  
    public static ThreadPoolProxy getSinglePool(String name) {  
        synchronized (mSingleLock) {  
            ThreadPoolProxy singlePool = mMap.get(name);  
            if (singlePool == null) {  
                singlePool = new ThreadPoolProxy(1, 1, 5L);  
                mMap.put(name, singlePool);  
            }  
            return singlePool;  
        }  
    }  
 
    public static class ThreadPoolProxy {  
        private ThreadPoolExecutor mPool;  
        private int mCorePoolSize;  
        private int mMaximumPoolSize;  
        private long mKeepAliveTime;  
 
        private ThreadPoolProxy(int corePoolSize, int maximumPoolSize, long keepAliveTime) {  
            mCorePoolSize = corePoolSize;  
            mMaximumPoolSize = maximumPoolSize;  
            mKeepAliveTime = keepAliveTime;  
        }  
 
        /** 执行任务,当线程池处于关闭,将会重新创建新的线程池 */  
        public synchronized void execute(Runnable run) {  
            if (run == null) {  
                return;  
            }  
            if (mPool == null || mPool.isShutdown()) {  
                mPool = new ThreadPoolExecutor(mCorePoolSize, mMaximumPoolSize, mKeepAliveTime, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>(), Executors.defaultThreadFactory(), new AbortPolicy());  
            }  
            mPool.execute(run);  
        }  
 
        /** 取消线程池中某个还未执行的任务 */  
        public synchronized void cancel(Runnable run) {  
            if (mPool != null && (!mPool.isShutdown() || mPool.isTerminating())) {  
                mPool.getQueue().remove(run);  
            }  
        }  
 
        /** 取消线程池中某个还未执行的任务 */  
        public synchronized boolean contains(Runnable run) {  
            if (mPool != null && (!mPool.isShutdown() || mPool.isTerminating())) {  
                return mPool.getQueue().contains(run);  
            } else {  
                return false;  
            }  
        }  
 
        /** 立刻关闭线程池,并且正在执行的任务也将会被中断 */  
        public void stop() {  
            if (mPool != null && (!mPool.isShutdown() || mPool.isTerminating())) {  
                mPool.shutdownNow();  
            }  
        }  
 
        /** 平缓关闭单任务线程池,但是会确保所有已经加入的任务都将会被执行完毕才关闭 */  
        public synchronized void shutdown() {  
            if (mPool != null && (!mPool.isShutdown() || mPool.isTerminating())) {  
                mPool.shutdownNow();  
            }  
        }  
 
    }  
}  

这个线程池工具类 主要就是 生成一个线程池, 以及 取消线程池中的任务,查询线程池中是否包含某一任务。

下载任务 DownloadTask

我们的现在线程 DownloadTask 就 通过 ThreadManager .getDownloadPool().execute() 方法 交给线程池去管理。

有了线程池管理我们的线程, 那我们下一步 就是 DownloadTask 这个类去下载了。

/** 下载任务 */  
public class DownloadTask implements Runnable {  
  private DownloadInfo info;  

  public DownloadTask(DownloadInfo info) {  
      this.info = info;  
  }  

  @Override  
  public void run() {  
      info.setDownloadState(STATE_DOWNLOADING);// 先改变下载状态  
      notifyDownloadStateChanged(info);  
      File file = new File(info.getPath());// 获取下载文件  
      HttpResult httpResult = null;  
      InputStream stream = null;  
      if (info.getCurrentSize() == 0 || !file.exists()  
              || file.length() != info.getCurrentSize()) {  
          // 如果文件不存在,或者进度为0,或者进度和文件长度不相符,就需要重新下载  

          info.setCurrentSize(0);  
          file.delete();  
      }  
      httpResult = HttpHelper.download(info.getUrl());  
      // else {  
      // // //文件存在且长度和进度相等,采用断点下载  
      // httpResult = HttpHelper.download(info.getUrl() + "&range=" +  
      // info.getCurrentSize());  
      // }  
      if (httpResult == null  
              || (stream = httpResult.getInputStream()) == null) {  
          info.setDownloadState(STATE_ERROR);// 没有下载内容返回,修改为错误状态  
          notifyDownloadStateChanged(info);  
      } else {  
          try {  
              skipBytesFromStream(stream, info.getCurrentSize());  
          } catch (Exception e1) {  
              e1.printStackTrace();  
          }  

          FileOutputStream fos = null;  
          try {  
              fos = new FileOutputStream(file, true);  
              int count = -1;  
              byte[] buffer = new byte[1024];  
              while (((count = stream.read(buffer)) != -1)  
                      && info.getDownloadState() == STATE_DOWNLOADING) {  
                  // 每次读取到数据后,都需要判断是否为下载状态,如果不是,下载需要终止,如果是,则刷新进度  
                  fos.write(buffer, 0, count);  
                  fos.flush();  
                  info.setCurrentSize(info.getCurrentSize() + count);  
                  notifyDownloadProgressed(info);// 刷新进度  
              }  
          } catch (Exception e) {  
              info.setDownloadState(STATE_ERROR);  
              notifyDownloadStateChanged(info);  
              info.setCurrentSize(0);  
              file.delete();  
          } finally {  
              IOUtils.close(fos);  
              if (httpResult != null) {  
                  httpResult.close();  
              }  
          }  

          // 判断进度是否和app总长度相等  
          if (info.getCurrentSize() == info.getAppSize()) {  
              info.setDownloadState(STATE_DOWNLOADED);  
              notifyDownloadStateChanged(info);  
          } else if (info.getDownloadState() == STATE_PAUSED) {// 判断状态  
              notifyDownloadStateChanged(info);  
          } else {  
              info.setDownloadState(STATE_ERROR);  
              notifyDownloadStateChanged(info);  
              info.setCurrentSize(0);// 错误状态需要删除文件  
              file.delete();  
          }  
      }  
      mTaskMap.remove(info.getId());  
  }  
}  

下载的原理 很简单,就是 通过 目标的URL 拿到流,然后写到本地。

因为下载在 run() 里面执行,这个DownloadTask 类 我们就看run() 方法的实现,所以 关键代码 就是下面一点点

fos = new FileOutputStream(file, true);  
  int count = -1;  
  byte[] buffer = new byte[1024];  
  while (((count = stream.read(buffer)) != -1)  
          && info.getDownloadState() == STATE_DOWNLOADING) {  
      // 每次读取到数据后,都需要判断是否为下载状态,如果不是,下载需要终止,如果是,则刷新进度  
      fos.write(buffer, 0, count);  
      fos.flush();  
      info.setCurrentSize(info.getCurrentSize() + count);  
      notifyDownloadProgressed(info);// 刷新进度  
  }  

这个在我们刚接触Java 的时候 肯定都写过了。 这就是往本地写数据的代码。所以run()方法中的 前面 就是拿到 stream 输入流, 以及 把file 创建出来。

刷新进度,状态

关于控制 button中text 显示 暂停 ,下载,还是进度,就靠 notifyDownloadProgressed(info);和 notifyDownloadStateChanged(info)两个方法, 这两个方法 实际上调用的是两个接口,只要我们在我们需要改变界面的类里 实现这两个接口,就可以接收到 包含最新信息的info对象。而我们在哪个类里改变button 上面 显示的文字呢? 当然是在 我们的adapter 里面了,大家都知道 是在 adapter 的getView() 方法里面 加载的每一条数据的布局。

那就一起看下是不是这样子呢?

public class RecommendAdapter extends BaseAdapter implements  
        DownloadManager.DownloadObserver {  
 
    ArrayList<AppInfo> list;  
    private List<ViewHolder> mDisplayedHolders;  
    private FinalBitmap finalBitmap;  
    private Context context;  
 
    public RecommendAdapter(ArrayList<AppInfo> list, FinalBitmap finalBitmap,  
            Context context) {  
        this.list = list;  
        this.context = context;  
        this.finalBitmap = finalBitmap;  
        mDisplayedHolders = new ArrayList<ViewHolder>();  
    }  
 
 
 
    public void startObserver() {  
        DownloadManager.getInstance().registerObserver(this);  
    }  
 
    public void stopObserver() {  
        DownloadManager.getInstance().unRegisterObserver(this);  
    }  
 
    @Override  
    public int getCount() {  
        return list.size();  
    }  
 
    @Override  
    public Object getItem(int position) {  
        return list.get(position);  
    }  
 
    @Override  
    public long getItemId(int position) {  
        return position;  
    }  
 
    @Override  
    public View getView(int position, View convertView, ViewGroup parent) {  
        final AppInfo appInfo = list.get(position);  
        final ViewHolder holder;  
 
        if (convertView == null) {  
            holder = new ViewHolder(context);  
        } else {  
            holder = (ViewHolder) convertView.getTag();  
        }  
        holder.setData(appInfo);  
        mDisplayedHolders.add(holder);  
        return holder.getRootView();  
    }  
 
    @Override  
    public void onDownloadStateChanged(DownloadInfo info) {  
        refreshHolder(info);  
    }  
 
    @Override  
    public void onDownloadProgressed(DownloadInfo info) {  
        refreshHolder(info);  
 
    }  
 
    public List<ViewHolder> getDisplayedHolders() {  
        synchronized (mDisplayedHolders) {  
            return new ArrayList<ViewHolder>(mDisplayedHolders);  
        }  
    }  
 
    public void clearAllItem() {  
        if (list != null){  
            list.clear();  
        }  
        if (mDisplayedHolders != null) {  
            mDisplayedHolders.clear();  
        }  
    }  
 
    public void addItems(ArrayList<AppInfo> infos) {  
        list.addAll(infos);  
    }  
 
    private void refreshHolder(final DownloadInfo info) {  
        List<ViewHolder> displayedHolders = getDisplayedHolders();  
        for (int i = 0; i < displayedHolders.size(); i++) {  
            final ViewHolder holder = displayedHolders.get(i);  
            AppInfo appInfo = holder.getData();  
            if (appInfo.getId() == info.getId()) {  
                AppUtil.post(new Runnable() {  
                    @Override  
                    public void run() {  
                        holder.refreshState(info.getDownloadState(),  
                                info.getProgress());  
                    }  
                });  
            }  
        }  
 
    }  
 
    public class ViewHolder {  
        public TextView textView01;  
        public TextView textView02;  
        public TextView textView03;  
        public TextView textView04;  
        public ImageView imageView_icon;  
        public Button button;  
        public LinearLayout linearLayout;  
 
        public AppInfo mData;  
        private DownloadManager mDownloadManager;  
        private int mState;  
        private float mProgress;  
        protected View mRootView;  
        private Context context;  
        private boolean hasAttached;  
 
        public ViewHolder(Context context) {  
            mRootView = initView();  
            mRootView.setTag(this);  
            this.context = context;  
 
 
        }  
 
        public View getRootView() {  
            return mRootView;  
        }  
 
        public View initView() {  
            View view = AppUtil.inflate(R.layout.item_recommend_award);  
 
            imageView_icon = (ImageView) view  
                    .findViewById(R.id.imageview_task_app_cion);  
 
            textView01 = (TextView) view  
                    .findViewById(R.id.textview_task_app_name);  
            textView02 = (TextView) view  
                    .findViewById(R.id.textview_task_app_size);  
            textView03 = (TextView) view  
                    .findViewById(R.id.textview_task_app_desc);  
            textView04 = (TextView) view  
                    .findViewById(R.id.textview_task_app_love);  
            button = (Button) view.findViewById(R.id.button_task_download);  
            linearLayout = (LinearLayout) view  
                    .findViewById(R.id.linearlayout_task);  
 
            button.setOnClickListener(new OnClickListener() {  
                @Override  
                public void onClick(View v) {  
                    System.out.println("mState:173    "+mState);  
                    if (mState == DownloadManager.STATE_NONE  
                            || mState == DownloadManager.STATE_PAUSED  
                            || mState == DownloadManager.STATE_ERROR) {  
 
                        mDownloadManager.download(mData);  
                    } else if (mState == DownloadManager.STATE_WAITING  
                            || mState == DownloadManager.STATE_DOWNLOADING) {  
                        mDownloadManager.pause(mData);  
                    } else if (mState == DownloadManager.STATE_DOWNLOADED) {  
//                      tell2Server();  
                        mDownloadManager.install(mData);  
                    }  
                }  
            });  
            return view;  
        }  
 
 
        public void setData(AppInfo data) {  
 
            if (mDownloadManager == null) {  
                mDownloadManager = DownloadManager.getInstance();  
 
            }  
             String filepath= FileUtil.getDownloadDir(AppUtil.getContext()) + File.separator + data.getName() + ".apk";  
 
                boolean existsFile = FileUtil.isExistsFile(filepath);  
                if(existsFile){  
                    int fileSize = FileUtil.getFileSize(filepath);  
 
                    if(data.getSize()==fileSize){  
                        DownloadInfo downloadInfo = DownloadInfo.clone(data);  
                        downloadInfo.setCurrentSize(data.getSize());  
                        downloadInfo.setHasFinished(true);  
                        mDownloadManager.setDownloadInfo(data.getId(),downloadInfo );  
                    }  
//                  else if(fileSize>0){  
//                      DownloadInfo downloadInfo = DownloadInfo.clone(data);  
//                      downloadInfo.setCurrentSize(data.getSize());  
//                      downloadInfo.setHasFinished(false);  
//                      mDownloadManager.setDownloadInfo(data.getId(),downloadInfo );  
//                  }  
 
                }  
 
            DownloadInfo downloadInfo = mDownloadManager.getDownloadInfo(data  
                    .getId());  
            if (downloadInfo != null) {  
 
                mState = downloadInfo.getDownloadState();  
                mProgress = downloadInfo.getProgress();  
            } else {  
 
                mState = DownloadManager.STATE_NONE;  
                mProgress = 0;  
            }  
            this.mData = data;  
            refreshView();  
        }  
 
        public AppInfo getData() {  
            return mData;  
        }  
 
        public void refreshView() {  
            linearLayout.removeAllViews();  
            AppInfo info = getData();  
            textView01.setText(info.getName());  
            textView02.setText(FileUtil.FormetFileSize(info.getSize()));  
            textView03.setText(info.getDes());  
            textView04.setText(info.getDownloadNum() + "下载量);  
            finalBitmap.display(imageView_icon, info.getIconUrl());  
 
 
            if (info.getType().equals("0")) {  
//              mState = DownloadManager.STATE_READ;  
                textView02.setVisibility(View.GONE);  
            }else{  
                String  path=FileUtil.getDownloadDir(AppUtil.getContext()) + File.separator + info.getName() + ".apk";  
                hasAttached = FileUtil.isValidAttach(path, false);  
 
                DownloadInfo downloadInfo = mDownloadManager.getDownloadInfo(info  
                        .getId());  
                if (downloadInfo != null && hasAttached) {  
                    if(downloadInfo.isHasFinished()){  
 
                        mState = DownloadManager.STATE_DOWNLOADED;  
                    }else{  
                        mState = DownloadManager.STATE_PAUSED;  
 
                    }  
 
                } else {  
                    mState = DownloadManager.STATE_NONE;  
                    if(downloadInfo !=null){  
                        downloadInfo.setDownloadState(mState);  
                    }  
                }  
            }  
 
            refreshState(mState, mProgress);  
        }  
 
        public void refreshState(int state, float progress) {  
            mState = state;  
            mProgress = progress;  
            switch (mState) {  
            case DownloadManager.STATE_NONE:  
                button.setText(R.string.app_state_download);  
                break;  
            case DownloadManager.STATE_PAUSED:  
                button.setText(R.string.app_state_paused);  
                break;  
            case DownloadManager.STATE_ERROR:  
                button.setText(R.string.app_state_error);  
                break;  
            case DownloadManager.STATE_WAITING:  
                button.setText(R.string.app_state_waiting);  
                break;  
            case DownloadManager.STATE_DOWNLOADING:  
                button.setText((int) (mProgress * 100) + "%");  
                break;  
            case DownloadManager.STATE_DOWNLOADED:  
                button.setText(R.string.app_state_downloaded);  
                break;  
//          case DownloadManager.STATE_READ:  
//              button.setText(R.string.app_state_read);  
//              break;  
            default:  
                break;  
            }  
        }  
    }  
}  

何时 注册 监听observer

里面代码有点多,那就看startObserver()方法做了什么。

public void startObserver() {  
       DownloadManager.getInstance().registerObserver(this);  
   }  

这里 是 注册了observer, Observer 是什么东西?在DownloadManager 中我们定义了

public interface DownloadObserver {

    public void onDownloadStateChanged(DownloadInfo info);

    public void onDownloadProgressed(DownloadInfo info);
}

一个接口,里面有两个抽象方法 一个是 进度,另一个是下载状态。
那回过头来,屡一下, 我们在 下载的关键代码里面调用了 DownloadObserver onDownloadProgressed() DownloadObserver.onDownloadStateChanged()两个抽象方法,而我们在 adapter

@Override  
  public void onDownloadStateChanged(DownloadInfo info) {  
      refreshHolder(info);  
  }  
 
  @Override  
  public void onDownloadProgressed(DownloadInfo info) {  
      refreshHolder(info);  
 
  }  

中实现了 这两个方法 就可以轻松的控制 去 刷新 和改变 下载状态了。

细心的朋友 或许 发现问题了,对,我们还没有注册Observer,就在 DownloadManager 中去调用了。
这里 在看下DownloadManager 中 调用的方法

/** 当下载状态发送改变的时候回调 */
public void notifyDownloadStateChanged(DownloadInfo info) {
    synchronized (mObservers) {
        for (DownloadObserver observer : mObservers) {
            observer.onDownloadStateChanged(info);
        }
    }
}

/** 当下载进度发送改变的时候回调 */
public void notifyDownloadProgressed(DownloadInfo info) {
    synchronized (mObservers) {
        for (DownloadObserver observer : mObservers) {
            observer.onDownloadProgressed(info);
        }
    }
}

是的,这里我们遍历一个observer 容器,然后去刷新 ,所以我们还需要 把 Observer 对象 添加到 集合 mObservers 中,

所以肯定有这样一个方法 讲 observer 添加到集合中 。
/* 注册观察者 /
public void registerObserver(DownloadObserver observer) {
synchronized (mObservers) {
if (!mObservers.contains(observer)) {
mObservers.add(observer);
}
}
}

/** 反注册观察者 */
public void unRegisterObserver(DownloadObserver observer) {
    synchronized (mObservers) {
        if (mObservers.contains(observer)) {
            mObservers.remove(observer);
        }
    }
}

所以最后一步,因为 adapter 方法中有 startObserver, 所以 我们在 主界面 MainActivity 的类中调用 adapter.startObser() 将 实现了 接口的adapter 对象 添加到 Observer 容器中 就可以了。

OK。大功告成!

=============================================

DownloadManager 代码

这里 贴一下DownloadManager 代码

public class DownloadManager {  
    public static final int STATE_NONE = 0;  
    /** 等待中 */  
    public static final int STATE_WAITING = 1;  
    /** 下载中 */  
    public static final int STATE_DOWNLOADING = 2;  
    /** 暂停 */  
    public static final int STATE_PAUSED = 3;  
    /** 下载完毕 */  
    public static final int STATE_DOWNLOADED = 4;  
    /** 下载失败 */  
    public static final int STATE_ERROR = 5;  
 
    // public static final int STATE_READ = 6;  
 
    private static DownloadManager instance;  
 
    private DownloadManager() {  
    }  
 
    /** 用于记录下载信息,如果是正式项目,需要持久化保存 */  
    private Map<Long, DownloadInfo> mDownloadMap = new ConcurrentHashMap<Long, DownloadInfo>();  
    /** 用于记录观察者,当信息发送了改变,需要通知他们 */  
    private List<DownloadObserver> mObservers = new ArrayList<DownloadObserver>();  
    /** 用于记录所有下载的任务,方便在取消下载时,通过id能找到该任务进行删除 */  
    private Map<Long, DownloadTask> mTaskMap = new ConcurrentHashMap<Long, DownloadTask>();  
 
    public static synchronized DownloadManager getInstance() {  
        if (instance == null) {  
            instance = new DownloadManager();  
        }  
        return instance;  
    }  
 
    /** 注册观察者 */  
    public void registerObserver(DownloadObserver observer) {  
        synchronized (mObservers) {  
            if (!mObservers.contains(observer)) {  
                mObservers.add(observer);  
            }  
        }  
    }  
 
    /** 反注册观察者 */  
    public void unRegisterObserver(DownloadObserver observer) {  
        synchronized (mObservers) {  
            if (mObservers.contains(observer)) {  
                mObservers.remove(observer);  
            }  
        }  
    }  
 
    /** 当下载状态发送改变的时候回调 */  
    public void notifyDownloadStateChanged(DownloadInfo info) {  
        synchronized (mObservers) {  
            for (DownloadObserver observer : mObservers) {  
                observer.onDownloadStateChanged(info);  
            }  
        }  
    }  
 
    /** 当下载进度发送改变的时候回调 */  
    public void notifyDownloadProgressed(DownloadInfo info) {  
        synchronized (mObservers) {  
            for (DownloadObserver observer : mObservers) {  
                observer.onDownloadProgressed(info);  
            }  
        }  
    }  
 
    /** 下载,需要传入一个appInfo对象 */  
    public synchronized void download(AppInfo appInfo) {  
        // 先判断是否有这个app的下载信息  
        DownloadInfo info = mDownloadMap.get(appInfo.getId());  
        if (info == null) {// 如果没有,则根据appInfo创建一个新的下载信息  
            info = DownloadInfo.clone(appInfo);  
            mDownloadMap.put(appInfo.getId(), info);  
        }  
        // 判断状态是否为STATE_NONE、STATE_PAUSED、STATE_ERROR。只有这3种状态才能进行下载,其他状态不予处理  
        if (info.getDownloadState() == STATE_NONE  
                || info.getDownloadState() == STATE_PAUSED  
                || info.getDownloadState() == STATE_ERROR) {  
            // 下载之前,把状态设置为STATE_WAITING,因为此时并没有产开始下载,只是把任务放入了线程池中,当任务真正开始执行时,才会改为STATE_DOWNLOADING  
            info.setDownloadState(STATE_WAITING);  
            notifyDownloadStateChanged(info);// 每次状态发生改变,都需要回调该方法通知所有观察者  
            DownloadTask task = new DownloadTask(info);// 创建一个下载任务,放入线程池  
            mTaskMap.put(info.getId(), task);  
            ThreadManager.getDownloadPool().execute(task);  
        }  
    }  
 
    /** 暂停下载 */  
    public synchronized void pause(AppInfo appInfo) {  
        stopDownload(appInfo);  
        DownloadInfo info = mDownloadMap.get(appInfo.getId());// 找出下载信息  
        if (info != null) {// 修改下载状态  
            info.setDownloadState(STATE_PAUSED);  
            notifyDownloadStateChanged(info);  
        }  
    }  
 
    /** 取消下载,逻辑和暂停类似,只是需要删除已下载的文件 */  
    public synchronized void cancel(AppInfo appInfo) {  
        stopDownload(appInfo);  
        DownloadInfo info = mDownloadMap.get(appInfo.getId());// 找出下载信息  
        if (info != null) {// 修改下载状态并删除文件  
            info.setDownloadState(STATE_NONE);  
            notifyDownloadStateChanged(info);  
            info.setCurrentSize(0);  
            File file = new File(info.getPath());  
            file.delete();  
        }  
    }  
 
    /** 安装应用 */  
    public synchronized void install(AppInfo appInfo) {  
        stopDownload(appInfo);  
        DownloadInfo info = mDownloadMap.get(appInfo.getId());// 找出下载信息  
        if (info != null) {// 发送安装的意图  
            Intent installIntent = new Intent(Intent.ACTION_VIEW);  
            installIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);  
            installIntent.setDataAndType(Uri.parse("file://" + info.getPath()),  
                    "application/vnd.android.package-archive");  
            AppUtil.getContext().startActivity(installIntent);  
        }  
        notifyDownloadStateChanged(info);  
    }  
 
    /** 启动应用,启动应用是最后一个 */  
    public synchronized void open(AppInfo appInfo) {  
        try {  
            Context context = AppUtil.getContext();  
            // 获取启动Intent  
            Intent intent = context.getPackageManager()  
                    .getLaunchIntentForPackage(appInfo.getPackageName());  
            context.startActivity(intent);  
        } catch (Exception e) {  
        }  
    }  
 
    /** 如果该下载任务还处于线程池中,且没有执行,先从线程池中移除 */  
    private void stopDownload(AppInfo appInfo) {  
        DownloadTask task = mTaskMap.remove(appInfo.getId());// 先从集合中找出下载任务  
        if (task != null) {  
            ThreadManager.getDownloadPool().cancel(task);// 然后从线程池中移除  
        }  
    }  
 
    /** 获取下载信息 */  
    public synchronized DownloadInfo getDownloadInfo(long id) {  
        return mDownloadMap.get(id);  
    }  
 
    public synchronized void setDownloadInfo(long id, DownloadInfo info) {  
        mDownloadMap.put(id, info);  
    }  
 
    /** 下载任务 */  
    public class DownloadTask implements Runnable {  
        private DownloadInfo info;  
 
        public DownloadTask(DownloadInfo info) {  
            this.info = info;  
        }  
 
        @Override  
        public void run() {  
            info.setDownloadState(STATE_DOWNLOADING);// 先改变下载状态  
            notifyDownloadStateChanged(info);  
            File file = new File(info.getPath());// 获取下载文件  
            HttpResult httpResult = null;  
            InputStream stream = null;  
            if (info.getCurrentSize() == 0 || !file.exists()  
                    || file.length() != info.getCurrentSize()) {  
                // 如果文件不存在,或者进度为0,或者进度和文件长度不相符,就需要重新下载  
 
                info.setCurrentSize(0);  
                file.delete();  
            }  
            httpResult = HttpHelper.download(info.getUrl());  
            // else {  
            // // //文件存在且长度和进度相等,采用断点下载  
            // httpResult = HttpHelper.download(info.getUrl() + "&range=" +  
            // info.getCurrentSize());  
            // }  
            if (httpResult == null  
                    || (stream = httpResult.getInputStream()) == null) {  
                info.setDownloadState(STATE_ERROR);// 没有下载内容返回,修改为错误状态  
                notifyDownloadStateChanged(info);  
            } else {  
                try {  
                    skipBytesFromStream(stream, info.getCurrentSize());  
                } catch (Exception e1) {  
                    e1.printStackTrace();  
                }  
 
                FileOutputStream fos = null;  
                try {  
                    fos = new FileOutputStream(file, true);  
                    int count = -1;  
                    byte[] buffer = new byte[1024];  
                    while (((count = stream.read(buffer)) != -1)  
                            && info.getDownloadState() == STATE_DOWNLOADING) {  
                        // 每次读取到数据后,都需要判断是否为下载状态,如果不是,下载需要终止,如果是,则刷新进度  
                        fos.write(buffer, 0, count);  
                        fos.flush();  
                        info.setCurrentSize(info.getCurrentSize() + count);  
                        notifyDownloadProgressed(info);// 刷新进度  
                    }  
                } catch (Exception e) {  
                    info.setDownloadState(STATE_ERROR);  
                    notifyDownloadStateChanged(info);  
                    info.setCurrentSize(0);  
                    file.delete();  
                } finally {  
                    IOUtils.close(fos);  
                    if (httpResult != null) {  
                        httpResult.close();  
                    }  
                }  
 
                // 判断进度是否和app总长度相等  
                if (info.getCurrentSize() == info.getAppSize()) {  
                    info.setDownloadState(STATE_DOWNLOADED);  
                    notifyDownloadStateChanged(info);  
                } else if (info.getDownloadState() == STATE_PAUSED) {// 判断状态  
                    notifyDownloadStateChanged(info);  
                } else {  
                    info.setDownloadState(STATE_ERROR);  
                    notifyDownloadStateChanged(info);  
                    info.setCurrentSize(0);// 错误状态需要删除文件  
                    file.delete();  
                }  
            }  
            mTaskMap.remove(info.getId());  
        }  
    }  
 
    public interface DownloadObserver {  
 
        public abstract void onDownloadStateChanged(DownloadInfo info);  
 
        public abstract void onDownloadProgressed(DownloadInfo info);  
    }  
 
    /* 重写了Inpustream 中的skip(long n) 方法,将数据流中起始的n 个字节跳过 */  
    private long skipBytesFromStream(InputStream inputStream, long n) {  
        long remaining = n;  
        // SKIP_BUFFER_SIZE is used to determine the size of skipBuffer  
        int SKIP_BUFFER_SIZE = 10000;  
        // skipBuffer is initialized in skip(long), if needed.  
        byte[] skipBuffer = null;  
        int nr = 0;  
        if (skipBuffer == null) {  
            skipBuffer = new byte[SKIP_BUFFER_SIZE];  
        }  
        byte[] localSkipBuffer = skipBuffer;  
        if (n <= 0) {  
            return 0;  
        }  
        while (remaining > 0) {  
            try {  
                long skip = inputStream.skip(10000);  
                nr = inputStream.read(localSkipBuffer, 0,  
                        (int) Math.min(SKIP_BUFFER_SIZE, remaining));  
            } catch (IOException e) {  
                e.printStackTrace();  
            }  
            if (nr < 0) {  
                break;  
            }  
            remaining -= nr;  
        }  
        return n - remaining;  
    }  
}  

有两点 需要说明,关于 点击暂停后,再继续下载 有两种方式可以实现

第一种 点击暂停的时候 记录下载了 多少,然后 再点击 继续下载 时,告诉服务器, 让服务器接着 上次的数据 往本地传递,

代码 就是 我们 DownloadTask 下载时候,判断一下

// //文件存在且长度和进度相等,采用断点下载  
httpResult = HttpHelper.download(info.getUrl() + "&range=" + info.getCurrentSize());  

通过 range 来区分 当前的下载size.

服务器 处理的代码 也很简单 就是一句话

String range = req.getParameter(“range”); 拿到 range 判断 range 存在不存在。 

如果不存在

FileInputStream stream = new FileInputStream(file);  
          int count = -1;  
          byte[] buffer = new byte[1024];  
          while ((count = stream.read(buffer)) != -1) {  
              SystemClock.sleep(20);  
              out.write(buffer, 0, count);  
              out.flush();  
          }  
          stream.close();  
          out.close();  

如果存在 那么跳过range 个字节

RandomAccessFile raf = new RandomAccessFile(file, "r");  
            raf.seek(Long.valueOf(range));    
            int count = -1;  
            byte[] buffer = new byte[1024];  
            while ((count = raf.read(buffer)) != -1) {  
                SystemClock.sleep(10);  
                out.write(buffer, 0, count);  
                out.flush();  
            }  
            raf.close();  
            out.close();  

另一种方式是本地处理,这个demo 中就是本地处理的, 但是有一个问题, 因为 Java api的原因 ,inputStream.skip() 方法 并不能准确的 跳过多少个字节,

而是 小于你想要跳过的字节,所以 你要去遍历 一直到 满足你要跳过的字节 在继续写, 因为 这样的方法有一个缺点,就是在下载很大的文件,

比如文件大小20M ,当已经下载了15M 此时你去暂停,在继续下载,那么要跳过前面的15M 将会话费很多时间。

所以这个仅限于学习。实际中 如果要下载大的文件,不能用这种方法。

Android 之多线程下载原理

在Android之中呢,对于多线程的操作很是平凡,所以对于多线程的理解越深,那么对于自己的程序便能够很好的运行

这也是对于Android开发是一个重要的知识点,那么我们现在来了解多线程的下载原理。

android 多线程下载
多线程下载步骤: 1.本地创建一个跟服务器一样的大小一样的文件 临时文件。 2.计算分配几个线程去下载服务器上的资源 每个文件下载的位置。 3.开启线程,每一个线程下载对应的文件。 4.如果所有的线程都把自己的数据下载完成了,服务器上的资源就被下载到本地了
如图所示:(假设有三个线程在进行下载) 

vcq9Cr+qyrzOu9bDo7oKo6jP37PMaWQgLSAxo6kgKiDDv9K7uPa/7LXEtPPQoQq94cr4zrvWw6O6IAqjqM/fs8xpZKOpKiDDv9K7v+y1xLTz0KEgLSAxCjxicj4KCs/Cw+ajrM7Sw8fPyNPDamF2YbT6wuvAtMq1z9bSu8/CCnBhY2thZ2UgY29tLnplbmd0YW8uZGVtbzs8YnI+Cjxicj4KPGJyPgppbXBvcnQgamF2YS5pby5JbnB1dFN0cmVhbTs8YnI+CmltcG9ydCBqYXZhLmlvLlJhbmRvbUFjY2Vzc0ZpbGU7PGJyPgppbXBvcnQgamF2YS5uZXQuSHR0cFVSTENvbm5lY3Rpb247PGJyPgppbXBvcnQgamF2YS5uZXQuVVJMOzxicj4KPGJyPgo8YnI+CnB1YmxpYyBjbGFzcyBEZW1vTG9hZGVyIHs8YnI+CnByaXZhdGUgc3RhdGljIERlbW9Mb2FkZXIgbG9hZGVyID0gbmV3IERlbW9Mb2FkZXIoKTs8YnI+CnByaXZhdGUgc3RhdGljIGludCB0aHJlYWRDb3VudCA9IDM7PGJyPgo8YnI+Cjxicj4KcHJpdmF0ZSBEZW1vTG9hZGVyKCkgezxicj4KPGJyPgo8YnI+Cn08YnI+Cjxicj4KLy8gtaXA/cnovMbEo8q9PGJyPgpwdWJsaWMgc3RhdGljIERlbW9Mb2FkZXIgZ2V0SW5zdGFuY2UoKSB7PGJyPgpyZXR1cm4gbG9hZGVyOzxicj4KfTxicj4KPGJyPgo8YnI+CnB1YmxpYyB2b2lkIGRvd25GaWxlKFN0cmluZyBwYXRoKSB7PGJyPgovLyDIpbf+zvHG97bLu/HIoc7EvP61xLOktsgs1NqxvrXYtLS9qNK7uPa4+rf+zvHG99K70fm089ChtcTOxLz+PGJyPgp0cnkgezxicj4KVVJMIHVybCA9IG5ldyBVUkwocGF0aCk7PGJyPgpIdHRwVVJMQ29ubmVjdGlvbiBjb25uZWN0aW9uID0gKEh0dHBVUkxDb25uZWN0aW9uKSB1cmw8YnI+Ci5vcGVuQ29ubmVjdGlvbigpOzxicj4KY29ubmVjdGlvbi5zZXREb0lucHV0KHRydWUpOzxicj4KY29ubmVjdGlvbi5zZXRSZXF1ZXN0TWV0aG9kKA=="GET");

connection.setReadTimeout(5000);
int code = connection.getResponseCode();
if (code == 200) {
// 获取服务器端文件的长度
int fileLength = connection.getContentLength();
// 本地创建一个跟服务器一样大小的文件
RandomAccessFile raf = new RandomAccessFile("setup.ext", "rwd");
raf.setLength(fileLength);
raf.close();
// 假设三个线程下载
int blockSize = fileLength / threadCount;
for (int threadId = 0; threadId < threadCount; threadId++) {
int startIndex = (threadId - 1) * blockSize;
int endIndex = threadId * blockSize - 1;
if (threadId == threadCount) {
endIndex = fileLength;
}
System.out.println("线程:" + threadId + ",下载:" + startIndex
+ "--->" + endIndex);
// 开始下载
new DownLoadThread(threadId, startIndex, endIndex, path)
.start();
}
System.out.println("文件总长度为:" + fileLength);
} else {
System.out.println("请求失败!");
}

} catch (Exception e) {
e.printStackTrace();
}
}

/**
* 下载文件的主线程
*
* @author Administrator zengtao
*
*/
public class DownLoadThread extends Thread {
private int threadId;
private int startIndex;
private int endIndex;
private String path;

/**
*
* @param threadId
* 线程id
* @param startIndex
* 线程下载开始位置
* @param endIndex
* 线程下载结束位置
* @param path
* 线程下载结束文件放置地址
*/
public DownLoadThread(int threadId, int startIndex, int endIndex,
String path) {
super();
this.threadId = threadId;
this.startIndex = startIndex;
this.endIndex = endIndex;
this.path = path;
}

@Override
public void run() {
super.run();
URL url;
try {
url = new URL(path);
HttpURLConnection connection = (HttpURLConnection) url
.openConnection();
// 请求服务器下载部分的文件,制定开始的位置,和结束位置
connection.setRequestProperty("Range", "bytes=" + startIndex
+ "-" + endIndex);
connection.setDoInput(true);
connection.setRequestMethod("GET");
connection.setReadTimeout(5000);
// 从服务器获取的全部数据,返回:200,从服务器获取部分数据,返回:206
int code = connection.getResponseCode();
System.out.println("code = " + code);
InputStream is = connection.getInputStream();
RandomAccessFile raf = new RandomAccessFile("setup.exe", "rwd");
// 随机写文件的时候,从什么时候开始
raf.seek(startIndex);
int len = 0;
byte[] buff = new byte[1024];
while ((len = is.read(buff)) != -1) {
raf.write(buff, 0, len);
}
is.close();
raf.close();
System.out.println("线程:" + threadId + ",下载完成");
} catch (Exception e) {
e.printStackTrace();
}
}
}
}

要使用该类的时候,只需要知道一个url地址,然后调用里面的downFile()方法,就会开始下载文件了,这样实现可以下载一个安装包,比如:在网上下载一个qq,微信等的安装包,自己安装到电脑上,便可以用该方法实现。

时间: 2024-09-13 09:01:12

Android开发仿下载助手多线程下载及原理的相关文章

android 开发环境怎么去哪里下载,为什么安卓官网没法下载

问题描述 android 开发环境怎么去哪里下载,为什么安卓官网没法下载 android 开发环境怎么去哪里下载,为什么安卓官网没法下载 解决方案 这里算是很全的了 http://www.androiddevtools.cn/ 解决方案二: 谷歌早就退出中国市场啦,所以没办法下载- 解决方案三: 去google官网下载的里面都有 解决方案四: 自己搜索一下,比如参考这里的http://332374363.blog.51cto.com/5262696/1310882,SDK正常访问不了,需要使用特

《Android应用开发与系统改造实战》——1.3节Android开发所需软件的下载

1.3 Android开发所需软件的下载Android应用开发与系统改造实战 1.3.1 Elcipse目前,Android官方已经给出的最新ADT集成开发环境的插件已经开始支持Eclipse的3.6(Helios)版本,也可以使用Eclipse3.4或者3.5版本.Eclipse的下载网址:http://www.eclipse.org/downloads/. 1.3.2 ADTADT是Eclipse的一个插件,全称为Android Development Tools.是Google开发用来给A

Android开发仿映客送礼物效果_Android

这里写链接内容仿映客送小礼物的特效,顺便复习一下属性动画,话不多说先看效果图. 需求分析 可以看到整个动画有几部分组成,那我们就把每个部分拆分出来各个击破. 1.要显示那些内容以及内容间的位置关系? 可以看到我们要显示用户头像,昵称,礼物图标以及数量.所以这里我选择用FrameLayout来作为根布局. 2.需要哪些动画以及动画的执行顺序? a.首先是整体从左到右飞入并有一个回弹(translationX + OvershootInterpolator) b.然后是礼物从左到右飞入而且是一个带减

Android开发仿扫一扫实现拍摄框内的照片功能_Android

就是仿照现在扫一扫的形式,周围是半透明的遮挡,然后中间是全透明的,拍摄后只截取框内的内容 查了很多博客,实现起来真的太复杂了,本人比较怕麻烦所以在很多地方偷懒了 先上效果图: 第一步:设置照相机预览以及拍照 这是所有步骤的前提,没有预览,用户怎么知道自己拍的什么呢.预览用的是SurfaceView 这篇博文写得已经十分详细了,打开照相机,然后拍照,而且十分简洁!不想别的博客一下就几百行代码不知所云.这篇代码可以复制下去当相机模版使用. 这里遇到一个问题,就是预览的效果是左转90度的,拍出来也是左

Android 开发仿简书登录框可删除内容或显示密码框的内容_Android

简书App 是我很喜欢的一款软件.今天就模仿了一下他的登录框.先上图: 好了下面上代码,自定义ImgEditText 继承与EditText.重写一些方法. package lyf.myimgedittextdemo; import android.content.Context; import android.graphics.Rect; import android.graphics.drawable.Drawable; import android.text.Editable; impor

Android开发仿QQ空间根据位置弹出PopupWindow显示更多操作效果_Android

我们打开QQ空间的时候有个箭头按钮点击之后弹出PopupWindow会根据位置的变化显示在箭头的上方还是下方,比普通的PopupWindow弹在屏幕中间显示好看的多. 先看QQ空间效果图: 这个要实现这个效果可以分几步进行 1.第一步自定义PopupWindow,实现如图的样式,这个继承PopupWindow自定义布局很容易实现 2.得到点击按钮的位置,根据位置是否在屏幕的中间的上方还是下方,将PopupWindow显示在控件的上方或者下方 3.适配问题,因为PopupWindow上面的操作列表

Android开发仿QQ空间根据位置弹出PopupWindow显示更多操作效果

我们打开QQ空间的时候有个箭头按钮点击之后弹出PopupWindow会根据位置的变化显示在箭头的上方还是下方,比普通的PopupWindow弹在屏幕中间显示好看的多. 先看QQ空间效果图: 这个要实现这个效果可以分几步进行 1.第一步自定义PopupWindow,实现如图的样式,这个继承PopupWindow自定义布局很容易实现 2.得到点击按钮的位置,根据位置是否在屏幕的中间的上方还是下方,将PopupWindow显示在控件的上方或者下方 3.适配问题,因为PopupWindow上面的操作列表

Android开发仿咸鱼键盘DEMO(修改版)

在这里布局我就不贴出来了 /** * 最终被调用的修改价格dialog */ protected void editPriceDialog() { // TODO Auto-generated method stub editPriceView = View.inflate(this, R.layout.dialog_price_input_keyboard, null); priceDialog = new Dialog(this, R.style.contactdialog); priceD

Android开发仿映客送礼物效果

这里写链接内容仿映客送小礼物的特效,顺便复习一下属性动画,话不多说先看效果图. 需求分析 可以看到整个动画有几部分组成,那我们就把每个部分拆分出来各个击破. 1.要显示那些内容以及内容间的位置关系? 可以看到我们要显示用户头像,昵称,礼物图标以及数量.所以这里我选择用FrameLayout来作为根布局. 2.需要哪些动画以及动画的执行顺序? a.首先是整体从左到右飞入并有一个回弹(translationX + OvershootInterpolator) b.然后是礼物从左到右飞入而且是一个带减