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()方法,這樣更有利於記憶體回收。