对于我目前的应用程序,我从不同的“事件中收集图像 提供者“在西班牙。
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
错误。
答案 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次,即使我将整个文件下载到缓冲区,然后将字节数组解码为位图。我尝试了以下内容,但没有一个适合我:
我不相信刷新的输入流解决方法也可以。 它似乎确实是一个Android skia bug。人们仍然在这里报告问题: http://code.google.com/p/android/issues/detail?id=6066。 除了如果位图为空重试,我没有解决此错误的方法,这只会减少获得错误的机会。