編寫android載入圖片的程式時,遇到了記憶體泄露問題!

來源:互聯網
上載者:User

Log如下:

04-04 18:16:32.774: E/dalvikvm-heap(30873): 10036224-byte external allocation too large for this process.04-04 18:16:32.805: E/GraphicsJNI(30873): VM won't let us allocate 10036224 bytes04-04 18:16:32.813: D/dalvikvm(30873): GC_FOR_MALLOC freed 0K, 51% free 6259K/12615K, external 22123K/24171K, paused 22ms04-04 18:16:32.813: D/skia(30873): --- decoder->decode returned false04-04 18:16:32.813: W/dalvikvm(30873): threadid=15: thread exiting with uncaught exception (group=0x40015568)04-04 18:16:32.821: E/AndroidRuntime(30873): FATAL EXCEPTION: AsyncTask #304-04 18:16:32.821: E/AndroidRuntime(30873): java.lang.RuntimeException: An error occured while executing doInBackground()04-04 18:16:32.821: E/AndroidRuntime(30873): at android.os.AsyncTask$3.done(AsyncTask.java:200)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:274)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.FutureTask.setException(FutureTask.java:125)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:308)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.FutureTask.run(FutureTask.java:138)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1088)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:581)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.lang.Thread.run(Thread.java:1019)04-04 18:16:32.821: E/AndroidRuntime(30873): Caused by: java.lang.OutOfMemoryError: bitmap size exceeds VM budget04-04 18:16:32.821: E/AndroidRuntime(30873): at android.graphics.BitmapFactory.nativeDecodeByteArray(Native Method)04-04 18:16:32.821: E/AndroidRuntime(30873): at android.graphics.BitmapFactory.decodeByteArray(BitmapFactory.java:419)04-04 18:16:32.821: E/AndroidRuntime(30873): at android.graphics.BitmapFactory.decodeByteArray(BitmapFactory.java:432)04-04 18:16:32.821: E/AndroidRuntime(30873): at com.jiadebin.ar.arnavigator.ARDetailActivity$wsAsyncTask.doInBackground(ARDetailActivity.java:220)04-04 18:16:32.821: E/AndroidRuntime(30873): at com.jiadebin.ar.arnavigator.ARDetailActivity$wsAsyncTask.doInBackground(ARDetailActivity.java:1)04-04 18:16:32.821: E/AndroidRuntime(30873): at android.os.AsyncTask$2.call(AsyncTask.java:185)04-04 18:16:32.821: E/AndroidRuntime(30873): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:306)04-04 18:16:32.821: E/AndroidRuntime(30873): ... 4 more

        我的程式是在一個activity裡用一個AsyncTask在後台載入Bitmap圖片(從網路伺服器上載入),這個activity被調用兩次之後就不能繼續調用了,第三次調用時就出現了這個問題然後程式異常退出,看Log裡說貌似是每一次調用載入線程之後,這個線程並沒有從記憶體中銷毀,而是繼續存在,所以導致後面再次建立該類線程時,記憶體泄露啦!

        在網上找了一下,遇到這類問題的人很多,通常的原因就是我們載入的圖片太大,我看了一下我的圖片,用500W像素的Defy拍的,原始大小一般都在800KB左右,這樣大的圖片才導致了記憶體泄露,網上大神們給的通常的解決方案是,用BitmapFactory解碼時加上設定好的options指令(我就是沒用options),通常Options如下設定:

BitmapFactory.Options options=new Options();
    options.inDither=false;    /*不進行圖片抖動處理*/
    options.inPreferredConfig=null;  /*設定讓解碼器以最佳方式解碼*/
    options.inSampleSize=4;          /*圖片長寬方向縮小倍數*/
    Bitmap img=BitmapFactory.decodeByteArray(buffer, 0, buffer.length, options);

這樣修改後,我發現傳到手機上的圖片一般都是200KB左右了,即縮小為原來的1/4,而且看起來還行,沒怎麼失真,幾乎不影響觀看效果,問題也解決了~

對了,最好在你使用完Bitmap之後,調用一下recycle()方法,這樣更有利於記憶體回收。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.