Android:用ThreadSafeClientConnManager下载图片的Bug

时间:2009-10-27 11:41:28

标签: java android thread-safety httpclient bugfender

对于我目前的应用程序,我从不同的“事件中收集图像 提供者“在西班牙。

  Bitmap bmp=null;
  HttpGet httpRequest = new HttpGet(strURL);

  long t = System.currentTimeMillis();
  HttpResponse response = (HttpResponse) httpclient.execute(httpRequest);
  Log.i(TAG, "Image ["+ strURL + "] fetched in [" + (System.currentTimeMillis()-t) + "ms]");

     HttpEntity entity = response.getEntity();
     InputStream instream = entity.getContent();
     bmp = BitmapFactory.decodeStream(instream);

     return bmp;

然而,当从salir.com下载图像时,我得到以下内容 logcat输出:

13970     Gallery_Activity  I  Fetching image 2/8 URL: http://media.salir.com/_images_/verticales/a/0/1/0/2540-los_inmortales_la_trattoria-marc_aureli_27_29_no.jpg
13970     ServiceHttpRequest  I  Image [http://media.salir.com/_images_/verticales/a/0/1/0/2540-los_inmortales_la_trattoria-marc_aureli_27_29_no.jpg] fetched in [146ms]
13970     skia  D  --- decoder->decode returned false

搜索该错误消息并没有提供太多有用的结果。

任何人都知道问题可能是什么?

格拉西亚斯!


更新1:

在再询问一下并测试不同的东西后,我发现问题似乎在于其他地方。即使我的logcat输出显示

13970     ServiceHttpRequest  I  Image [http://media.salir.com/_images_/verticales/a/0/1/0/2540-los_inmortales_la_trattoria-marc_aureli_27_29_no.jpg] fetched in [146ms]
getContentLength(): 93288

这是图像的正确长度,以字节为单位,似乎流或HTTP连接有问题。

我的原始代码(上图)正在利用ThreadSafeClientConnManager。如果我用一个简单的URLConnection替换它就可以完美地运行:

URL url = new URL(strURL);
URLConnection conn = url.openConnection();
conn.connect();
InputStream instream = conn.getInputStream();
bmp = BitmapFactory.decodeStream(instream);

所以,我现在想知道为什么我的ThreadSafeClientConnManager与我的所有其他联系(主要是交换JSONObjects)完美地工作(至少看起来确实如此),但不是来自某些特定网站的图像(例如salir.com - 对于大多数其他网站,它的工作原理)。是否存在我缺少的HTTP参数?

我目前的设置是:

HttpParams parameters = new BasicHttpParams();
HttpProtocolParams.setVersion(parameters, HttpVersion.HTTP_1_1);
HttpProtocolParams.setContentCharset(parameters, HTTP.UTF_8);
HttpProtocolParams.setUseExpectContinue(parameters, false); // some webservers have problems if this is set to true
ConnManagerParams.setMaxTotalConnections(parameters, MAX_TOTAL_CONNECTIONS);
HttpConnectionParams.setConnectionTimeout(parameters, CONNECTION_TIMEOUT);
HttpConnectionParams.setSoTimeout(parameters, SOCKET_TIMEOUT);

SchemeRegistry schReg = new SchemeRegistry();
schReg.register(new Scheme("http", 
     PlainSocketFactory.getSocketFactory(), HTTP_PORT));

ClientConnectionManager conMgr = new ThreadSafeClientConnManager(parameters,schReg);

DefaultHttpClient http_client = new DefaultHttpClient(conMgr, parameters);

更新2:

现在,奇怪的是,它确实适用于ThreadSafeClientConnManager - 有时 - 。如果我继续尝试下载图像并连续解码几次,它可能会在15-30次试验后起作用。很奇怪。

我希望有一个解决方案,因为我更喜欢使用ThreadSafeClientConnManager代替URLConnection


更新3:

正如Mike Mosher在下面的建议,似乎使用BufferedHttpEntity 解码错误不再出现。但是现在,即使比以前少,我也会遇到SkImageDecoder::Factory returned null错误。

11 个答案:

答案 0 :(得分:27)

Stefan,

我遇到了同样的问题,并没有找到太多的互联网搜索。很多人都有这个问题,但解决它的答案却很多。

我使用URLConnection获取图像,但我发现问题不在于下载,但是BitmapFactory.decodeStream在解码图像时遇到了问题。

我更改了代码以反映原始代码(使用httpRequest)。我做了一个更改,我在http://groups.google.com/group/android-developers/browse_thread/thread/171b8bf35dbbed96/c3ec5f45436ceec8?lnk=raot找到了(感谢Nilesh)。你需要添加“BufferedHttpEntity bufHttpEntity = new BufferedHttpEntity(entity);”

这是我以前的代码:

        conn = (HttpURLConnection) bitmapUrl.openConnection(); 
        conn.connect();
        is = conn.getInputStream();
        //bis = new BufferedInputStream(is);
        //bm = BitmapFactory.decodeStream(bis);
        bm = BitmapFactory.decodeStream(is);

她的代码是有效的:

            HttpGet httpRequest = null;

    try {
        httpRequest = new HttpGet(bitmapUrl.toURI());
    } catch (URISyntaxException e) {
        e.printStackTrace();
    }

    HttpClient httpclient = new DefaultHttpClient();
        HttpResponse response = (HttpResponse) httpclient.execute(httpRequest);

        HttpEntity entity = response.getEntity();
        BufferedHttpEntity bufHttpEntity = new BufferedHttpEntity(entity); 
        InputStream instream = bufHttpEntity.getContent();
        bm = BitmapFactory.decodeStream(instream);

正如我所说,我有一个下载大约40张图片的页面,您可以刷新以查看最新的照片。我几乎有一半失败了“decoder-> decode返回错误错误”。使用上面的代码,我没有遇到任何问题。

由于

答案 1 :(得分:5)

我的解决方案是不使用BitmapFactory.decodeStream()导致我看待它的方式,它使用SKIA解码器,有时似乎有些不稳定。你可以尝试这样的事情。

    Bitmap bitmap = null;
    InputStream in = null;
    BufferedOutputStream out = null;
    try {
            in = new BufferedInputStream(new URL(url).openStream(),
                     IO_BUFFER_SIZE);

            final ByteArrayOutputStream dataStream = new ByteArrayOutputStream();
            out = new BufferedOutputStream(dataStream, IO_BUFFER_SIZE);
            copy(in, out);
            out.flush();

            final byte[] data = dataStream.toByteArray();
            bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);


   } catch (......){

   }        

和copy()函数

private static void copy(InputStream in, OutputStream out) throws IOException {
    byte[] b = new byte[IO_BUFFER_SIZE];
    int read;
    while ((read = in.read(b)) != -1) {
        out.write(b, 0, read);
    }
}

IO_BUFFER_SIZE是一个常数,值为4 * 1024。

答案 2 :(得分:3)

这是一个Android错误。你的代码还可以。 “Android的解码器目前不支持部分解码数据。” 如果你在实体中得到图像可能它是一个部分输入流,而android现在无法应对。

解决方案是使用此线程中的FlushedInputStream: http://code.google.com/p/android/issues/detail?id=6066

答案 3 :(得分:2)

同时尝试添加此选项。

opt.inPurgeable = true;

答案 4 :(得分:2)

HttpGet httpRequest = null;
try {
  httpRequest = new HttpGet(url);
} catch (Exception e) {
  e.printStackTrace();
}
HttpClient httpclient = new DefaultHttpClient();
HttpResponse response = (HttpResponse) httpclient.execute(httpRequest);
HttpEntity entity = response.getEntity();
BufferedHttpEntity bufHttpEntity = new BufferedHttpEntity(entity); 
InputStream instream = bufHttpEntity.getContent();
Bitmap bm = BitmapFactory.decodeStream(instream);

答案 5 :(得分:0)

我遇到了类似的问题,根本问题是由于请求某些图片时超时。我转而使用Background Image Loader并且从那时起就没有问题了。希望这有帮助

答案 6 :(得分:0)

我现在尝试了树不同的方法,你使用简单的URLConnection,这似乎对你有用,Mike Mosher使用的方式也是这样[http://asantoso.wordpress.com/2008/03/07/download-and-view-image-from-the-web/]它们都导致解码返回false。 / p>

但是,如果我将图像转换为PNG,所有方法都可以正常工作!然而,我尝试将图像放在我的工作ftp主页上,然后所有jpgs都可以使用简单的解决方案加载......所以出于某种原因,似乎服务器速度不够快,它会在实际完全下载之前解析jpg文件什么的。

答案 7 :(得分:0)

尝试从字节数组解码图像时遇到了完全相同的问题。经过一些实验,解决方案似乎是在BitmapFactory的Options中分配一些临时存储。尝试:

Options options = new Options();
options.inTempStorage = new byte[256];
Bitmap newMapBitmap = BitmapFactory.decodeStream(instream, null, options);

如果问题没有立即解决,请尝试增加临时数组的大小。我认为大型位图文件需要更大的缓冲区来进行解码。

答案 8 :(得分:0)

在Options中设置默认缓冲区确实有助于解决与BitmapFactory有关的一些内存不足问题,但肯定不会涵盖所有这些问题。检索位图或位图标头时,基本问题似乎是某种超时。我现在正在将流写入临时文件,然后将临时文件传递给BitmapFactory,该工作正常。

答案 9 :(得分:0)

HttpURLConnection hConn = null;
        hConn = openHttpConnection(szUrl);

        hConn.setRequestMethod("GET");  
        hConn.setRequestProperty("User-Agent",szUserAgent);
        hConn.setRequestProperty("Accept",szAccept);
        hConn.setRequestProperty("Accept-Charset",szCharset);

        hConn.setInstanceFollowRedirects(true);
        hConn.setUseCaches(true);
        hConn.setChunkedStreamingMode(8*1024);
        hConn.setDoInput(true);
        hConn.setConnectTimeout(60*1000);
        hConn.setReadTimeout(60*1000);
        hConn.connect();


        InputStream bmpIs = hConn.getInputStream();
        BufferedInputStream bmpBis = new BufferedInputStream(bmpIs); 
        Bitmap bmpThumb = null;

        BitmapFactory.Options bfOpt = new BitmapFactory.Options();

        bfOpt.inScaled = true;
        bfOpt.inSampleSize = 2;
        bfOpt.inPurgeable = true;

        bmpThumb = BitmapFactory.decodeStream(bmpBis,null,bfOpt);

        if(bmpThumb == null)
        {
            for(int i=0; i<10; i++)
            {
                Thread.sleep(200);
                System.gc();

                bmpThumb = BitmapFactory.decodeStream(bmpBis,null,bfOpt);


                if(bmpThumb == null)
                    bfOpt.inSampleSize += 1;
                else
                    break;
            }
        }

答案 10 :(得分:0)

我遇到此错误“SkImageDecoder :: Factory返回null”大约8次中的1次,即使我将整个文件下载到缓冲区,然后将字节数组解码为位图。我尝试了以下内容,但没有一个适合我:

  • 仅使用网址,而不是http连接类型
  • 缓冲读者
  • 下载整个文件,然后解码

我不相信刷新的输入流解决方法也可以。 它似乎确实是一个Android skia bug。人们仍然在这里报告问题:  http://code.google.com/p/android/issues/detail?id=6066。 除了如果位图为空重试,我没有解决此错误的方法,这只会减少获得错误的机会。