zoukankan      html  css  js  c++  java
  • Android webview通过http get下载文件下载两次的问题及解决方法

    一、现象

       一般通过Android webview进行下载文件的方法是

       1.重写DownloadListener的onDownloadStart方法,在onDownloadStart方法中弹出对话框提示用户有新的文件需要下载

       2.用户点击确定之后,通过http get下载文件

       由于Android webview的实现,以上的下载文件步骤涉及到了两次get的操作。第一次是用户在webview中点击下载链接时,webview自动发送http get请求,这个时候服务器除了将文件信息发送过来之外,会同时将文件的内容发送给webview。第二次是在步骤2,由自己设计的程序发起的。

       为了验证如上结论,我在Android 4.4系统中的自带浏览器通过访问并下载这个测试链接,并用wireshark进行抓包查看结果。通过如下三张图,我觉得可以验证同一份文件确实被传了两次。因为两个不同http get请求之后都可以看到服务器向客户端发送的连续的TCP数据包。

    二、解决方法

       针对这个问题,我觉得比较完美的解决办法是webview在进行get的时候能够缓存相应的文件内容,并且开放相应的接口给应用进行缓存的读取。但是我并没有发现相应的api。

       于是便有了如下的解决办法:在webview加载url之前,通过链接中的一些特定字段发现进行下载操作。然后通过http head请求获取文件名和大小等相关信息。由于http head只是请求文件相关信息,所以服务器只是通过http response将文件信息返回而不通过tcp发送文件具体内容。

       2.1 实现代码

      在继承自WebViewClient的类的shouldOverrideUrlLoading方法中检查链接类型,并通过http head获取文件大小,名称信息。在以下代码基础上需要增加合适的获取文件名称代码和handler的代码。

    		if (url.contains("可能会进行下载的链接字段")) {
    			new Thread() {
    				public void run() {
    					HttpHead httpGet = new HttpHead(mUrl);
    					boolean success = true;
    					String filename="";
    					Integer fileLength=100;					
    					try {
    						HttpParams httpParameters = new BasicHttpParams();
    			        	HttpConnectionParams.setConnectionTimeout(httpParameters, 8000);
    			        	HttpConnectionParams.setSoTimeout(httpParameters, 8000);
    						DefaultHttpClient httpClient = new DefaultHttpClient(httpParameters);
    //需要的话可能得增加cookie
    HttpResponse httpResponse = httpClient.execute(httpGet); if (httpResponse.getStatusLine().getStatusCode() == 200) {                   //如果文件名保存在Content-Disposition中,并且用的是gb2312编码方式可以参考这段代码 String nameString=httpResponse.getFirstHeader("Content-Disposition").getValue(); char [] nameChar=new char[nameString.length()]; byte [] nameBytes=new byte[nameString.length()]; nameString.getChars(0, nameString.length(), nameChar, 0); for (int i = 0; i < nameChar.length; i++) { nameBytes[i]=(byte)nameChar[i]; } filename=EncodingUtils.getString(nameBytes, "gb2312"); Log.d(TAG, "file name is "+filename); String contentLength=httpResponse.getFirstHeader("Content-Length").getValue(); fileLength=Integer.parseInt(contentLength); success=true; }else { success=false; } } catch (Exception e) { success = false; e.printStackTrace(); } Message msg = Message.obtain(); if(success) { msg.what = GET_FILE_INFO_FINISHED; Bundle mBundle=new Bundle(); mBundle.putString("fileName", filename); mBundle.putInt("fileLength", fileLength); msg.setData(mBundle); } else { msg.what = GET_FILE_INFO_FAILED; } mHandler.sendMessage(msg); } }.start(); }else { view.loadUrl(url); }

       2.2 适用情况

       在Android 4.4中,根据网页的特点,可能会出现点击下载链接,而shouldOverrideUrlLoading不被调用的情况,这时2.1中方法则失效了。不过对于4.4之前的系统应该是适用的。

    三、参考材料

    3.1 http://stackoverflow.com/questions/11801787/webview-cant-download-file-without-requesting-it-twice

    3.2 http://stackoverflow.com/questions/12535414/is-setdownloadlistener-ondownloadstart-called-after-the-webview-already-gets-the

  • 相关阅读:
    Java连载91-Map常用方法、Hashtable、SortedMap
    Python爬虫连载11-cookie、session、验证SSL证书、数据提取简介
    Java连载90-Sorted、Map讲解
    HTML连载71-翻转菜单练习
    Java连载89-SorteSet、Comparable接口
    Python爬虫连载10-Requests模块、Proxy代理
    Java连载88-HashSet集合与hashCode方法重写
    [设计模式] 设计模式课程(五)--装饰模式
    [设计模式] 设计模式课程(四)-- 观察者模式
    [设计模式] 设计模式课程(二)-- 模板模式
  • 原文地址:https://www.cnblogs.com/jiangz/p/3722476.html
Copyright © 2011-2022 走看看