6

我有 ViewPager 用于显示可缩放图像(使用ImageViewTouch)。我需要从 Internet (http) 加载大型位图。大我的意思是2000x1000。图像需要很大,因为它们是可缩放的并且需要显示细节。服务器上的图像是 .jpg 格式,但这没问题 - 我可以更改它。

我怎样才能设法将这么大的图像加载到 ImageViewTouch (ImageView) 而不会出现内存问题?

现在我只使用这个(AsyncTask):

    ImageView currentView; //ImageView where to place image loaded from Internet
    String ImageUrl; //URL of image to load

    protected Bitmap doInBackground(ArrayList... params) {

            Bitmap bitmap;
            InputStream in = null;
            imageUrl = (String)params[0].get(1);

            try{
                HttpClient httpclient = new DefaultHttpClient();
                HttpResponse response = httpclient.execute(new HttpGet(imageUrl));
                in = response.getEntity().getContent();
            } catch(Exception e){
                e.printStackTrace();
            }
            try {
                bitmapa = BitmapFactory.decodeStream(in);
                in.close();
            } catch (IOException e1) {
                e1.printStackTrace();
            }

            return bitmap;
        }


        @Override
        protected void onPostExecute(Bitmap bitmap) {

            currentView.setImageBitmap(bitmap);
        }

它会导致许多内存问题:

E/dalvikvm-heap(369): 69560740-byte external allocation too large for this process.
E/GraphicsJNI(369): VM won't let us allocate 69560740 bytes

或者

E/AndroidRuntime(369): java.lang.RuntimeException: An error occured while executing doInBackground()
E/AndroidRuntime(369):  at android.os.AsyncTask$3.done(AsyncTask.java:200)
E/AndroidRuntime(369):  at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:274)
E/AndroidRuntime(369):  at java.util.concurrent.FutureTask.setException(FutureTask.java:125)
E/AndroidRuntime(369):  at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:308)
E/AndroidRuntime(369):  at java.util.concurrent.FutureTask.run(FutureTask.java:138)
E/AndroidRuntime(369):  at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1088)
E/AndroidRuntime(369):  at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:581)
E/AndroidRuntime(369):  at java.lang.Thread.run(Thread.java:1019)
E/AndroidRuntime(369): Caused by: java.lang.OutOfMemoryError: bitmap size exceeds VM budget
E/AndroidRuntime(369):  at android.graphics.BitmapFactory.nativeDecodeStream(Native Method)
E/AndroidRuntime(369):  at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:470)
E/AndroidRuntime(369):  at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:525)
E/AndroidRuntime(369):  at com.package.app.ImageDownloader.doInBackground(ImageDownloader.java:78)
E/AndroidRuntime(369):  at com.package.app.ImageDownloader.doInBackground(ImageDownloader.java:1)
E/AndroidRuntime(369):  at android.os.AsyncTask$2.call(AsyncTask.java:185)
E/AndroidRuntime(369):  at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:306)
E/AndroidRuntime(369):  ... 4 more
4

2 回答 2

2

您在哪个版本的操作系统上进行测试?在早期版本中,可用的堆空间远低于更高版本。

我会认真考虑缩小你的位图以避免这种情况。如果您正在为 mdpi 手机下载 2000x1000 Bitmap,那么这可能是个坏主意。

我还建议阅读这篇文章,以获取有关如何ImageView根据其尺寸准确地为特定加载适当图像的更多信息。

最后,您应该始终关闭InputStream一个finally块:

try {
        bitmap = BitmapFactory.decodeStream(in);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        if (in != null) { in.close(); }
    }

您还android:largeHeap="true"可以考虑使用专门为处理大型位图的应用程序(例如照片编辑应用程序)而设计的应用程序。

于 2013-02-03T16:18:25.440 回答
2

2000x1000 像素的图像很大,但不是那么大。

我明白了,虽然该进程试图在死前分配 69560740 个字节,但大约是 66MByte!

在最坏的情况下,您的 2000x1000 约为 2000x1000x4 = 8MBytes,远小于 66MBytes。其他事情正在发生。

此外,我发现在 AsyncTasks 中使用位图作为返回/结果值存在问题。已完成的 AsyncTask 仍会保留其结果,直到它们被垃圾收集。由于您不控制 AsyncTask 实例的垃圾收集,因此不要使用 Bitmap 作为返回/结果值。而是将 Bitmap 放在一个字段中,并在 doInBackground 即将结束时分配它,并使用此字段在 onPostExecute 中获取 Bitmap 并(!)将此字段设置为 null:

Bitmap bitmap;
protected Void doInBackground(ArrayList... params) {

        InputStream in = null;
        imageUrl = (String)params[0].get(1);

        try{
            HttpClient httpclient = new DefaultHttpClient();
            HttpResponse response = httpclient.execute(new HttpGet(imageUrl));
            in = response.getEntity().getContent();
        } catch(Exception e){
            e.printStackTrace();
        }
        try {
            bitmap = BitmapFactory.decodeStream(in);
            in.close();
        } catch (IOException e1) {
            e1.printStackTrace();
        }

        return null;
    }


    @Override
    protected void onPostExecute(Void unused) {
        if (bitmap != null) {
            currentView.setImageBitmap(bitmap);
        }

        // And don't forget to null the bitmap field!
        bitmap = null; 
    }
于 2013-02-06T00:55:06.487 回答