출처: http://blog.naver.com/nimbusob/147298809
** 결론부터 말하자면,
Android Activity 의 lifecycle 과, VM의 Object lifecycle 은 별개로 동작하기 때문에 발생하는 문제라 볼 수 있다.
VM 의 GC는 Object 가 죽은 상태(멤버의 참조 횟수가 0이고, 자신의 참조 횟수가 0일때)일 때만 수행되는데,
Activity 가 죽었다 하여 Object 가 죽은 것은 아니기 때문에 Bitmap 에 대한 참조가 남아 있어 GC를 하지 못하는 것이다.
이는 좀더 유연한 프로그래밍을 가능하게 하는 안드로이드 개발팀의 배려라 볼 수도 있겠지만,
어차피 Activity 코딩시에는 Activity 생명주기를 따라 코딩하는게 일반적이니 Android framework 에서 이정도는 강제로 해 주는게
어떨까 싶다. 아니면 Manifest 에 옵션을 넣어 주던가.
** 20130121 추가
Android 4.0 이상부터는 Bitmap이 Native heap 대신 Dalvik VM Heap 에 할당되기 때문에, 일반 Java 객체처럼 참조를 끊는 것이
가능해졌다. 따라서 GC 타이밍을 좀 더 정확하게 예측 가능하다. 즉, Bitmap.recycle() 같은 모호한 표현 말고, bitmap = null 과
같은 코드만으로도 자원을 반환할 수 있다.
아래와 같은 간단한 Activity 를 실행해 보자. 단 22라인의 이미지 파일의 해상도는 3888x2592 (1000만화소) 이며(JPEG Size 2.2MiB)
정확히 20155392 바이트 (19.22MiB) 의 메모리를 필요로 한다. 실행 기기는 Nexus S (버전 2.3.2) 로 32MiB 의 가용 메모리(heap)를
가지고 있기 때문에 문제없이 화면이 뜰 것이다.
31라인의 imageView.setImageBitmap(
null
);
를 주목하자.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 | package net.oomtest; import android.app.Activity; import android.graphics.BitmapFactory; import android.os.Bundle; import android.util.Log; import android.widget.ImageView; import android.widget.LinearLayout; public class OomtestActivity extends Activity { LinearLayout mainLayout ; ImageView imageView ; @Override protected void onCreate(Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout. main ); mainLayout = (LinearLayout) findViewById(R.id. mainLayout ); imageView = new ImageView( this ); imageView .setImageBitmap(BitmapFactory.decodeFile( "/mnt/sdcard/hwan.documents/images/bigimage/1_portrait.jpg" )); mainLayout .addView( imageView ); } @Override protected void onDestroy() { Log.d( "OOMTEST" , onDestroy" ); imageView .setImageBitmap( null ); super .onDestroy(); } } |
앱을 실행한 결과 화면은 아래와 같다.
아래는 앱 실행 직후 메모리 사용 상황이다. allocated: 부분을 유심히 보길 바란다.
(native 23745kB, dakvik 2763kB, total 26508kB)
Applications Memory Usage (kB): Uptime: 22510282 Realtime: 28703998 ** MEMINFO in pid 8286 [net.oomtest] ** native dalvik other total size: 23764 5379 N/A 29143 allocated: 23745 2763 N/A 26508 free: 18 2616 N/A 2634 (Pss): 602 137 22165 22904 (shared dirty): 2208 1848 7276 11332 (priv dirty): 508 56 20752 21316 Objects Views: 0 ViewRoots: 0 AppContexts: 0 Activities: 0 Assets: 2 AssetManagers: 2 Local Binders: 5 Proxy Binders: 10 Death Recipients: 0 OpenSSL Sockets: 0 SQL heap: 0 MEMORY_USED: 0 PAGECACHE_OVERFLOW: 0 MALLOC_SIZE: 0 Asset Allocations zip:/data/app/net.oomtest-1.apk:/resources.arsc: 1K |
19MB의 대용량 이미지가 별 탈 없이 뜨는게 놀랍다. 그렇다면 이제 화면을 회전시켜 보도록 한다.
안드로이드의 Activity 는 onPause() → onStop() → onDestroy() 의 생명주기를 타며 화면이 새로 그려질 때,
onCreate() → onStart() → onResume() 의 순으로 Activity 가 다시 시작된다.
화면방향이 바뀌면 현재 화면의 내용을 모두 버리고 새로운 화면을 그리게 되므로 우리의 코드는 설명한 모든 생명주기를 다 타게 된다.
(여담이지만 이로 인해 화면 방향전환이 꽤 느리며 그것을 회피할 수 있는 방법은 http://developer.android.com/guide/topics/resources/runtime-changes.html 의 'Handling Configuration Change Yourself' 섹션 및 http://developer.android.com/resources/articles/faster-screen-orientation-change.html 를 참고하면 된다. 이 글의 주제인 OOM 과는 무관한 내용이므로 생략)
화면 방향을 바꾸면 아마 아래와 같이 Exception 이 발생하며 앱이 죽을 것이다.
01-02 21:52:22.515: D/OOMTEST(9993): onDestroy
01-02 21:52:22.585: D/dalvikvm(9993): GC_EXTERNAL_ALLOC freed 40K, 49% free 2760K/5379K, external 21308K/23356K, paused 63ms
01-02 21:52:22.601: E/dalvikvm-heap(9993): 20155392-byte external allocation too large for this process.
01-02 21:52:22.613: E/GraphicsJNI(9993): VM won't let us allocate 20155392 bytes
01-02 21:52:22.624: D/dalvikvm(9993): GC_FOR_MALLOC freed <1K, 49% free 2760K/5379K, external 1625K/21308K, paused 14ms
01-02 21:52:22.624: D/skia(9993): --- decoder->decode returned false
01-02 21:52:22.628: D/AndroidRuntime(9993): Shutting down VM
01-02 21:52:22.628: W/dalvikvm(9993): threadid=1: thread exiting with uncaught exception (group=0x40015560)
01-02 21:52:22.628: E/AndroidRuntime(9993): FATAL EXCEPTION: main
01-02 21:52:22.628: E/AndroidRuntime(9993): java.lang.OutOfMemoryError: bitmap size exceeds VM budget
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.graphics.BitmapFactory.nativeDecodeStream(Native Method)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:470)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.graphics.BitmapFactory.decodeFile(BitmapFactory.java:284)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.graphics.BitmapFactory.decodeFile(BitmapFactory.java:309)
01-02 21:52:22.628: E/AndroidRuntime(9993): at net.oomtest.OomtestActivity.onCreate(OomtestActivity.java:24)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1611)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1663)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:2832)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.ActivityThread.access$1600(ActivityThread.java:117)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:935)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.os.Handler.dispatchMessage(Handler.java:99)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.os.Looper.loop(Looper.java:130)
01-02 21:52:22.628: E/AndroidRuntime(9993): at android.app.ActivityThread.main(ActivityThread.java:3683)
01-02 21:52:22.628: E/AndroidRuntime(9993): at java.lang.reflect.Method.invokeNative(Native Method)
01-02 21:52:22.628: E/AndroidRuntime(9993): at java.lang.reflect.Method.invoke(Method.java:507)
01-02 21:52:22.628: E/AndroidRuntime(9993): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
01-02 21:52:22.628: E/AndroidRuntime(9993): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
01-02 21:52:22.628: E/AndroidRuntime(9993): at dalvik.system.NativeStart.main(Native Method)
역시 예상대로 OOM Error 다.
그런데 왜 화면전환시 onDestroy() 에서 imageView.setImageBitmap(
null
);
를 호출해 주었는데도 OOM 이 발생할까?
첫 화면에서 사용한 bitmap 은 화면전환 이후 더 이상 필요없으므로, 첫 화면의 onDestroy() 호출 이후에 참조가 제거되어야 함에도
제거되지 않아 VM 이 GC를 제대로 해 주지 못해 발생하는 현상이다. 프레임웍은 지금같은 상황은 화면(Activity)이 바뀌지 않았으므로
이미지를 버릴 필요가 없다고 판단한 모양이다. 허나 첫 화면이 onDestroy 로 사라졌으면 그 화면 내에서 선언한 내용들은 GC 대상이
되어야 함에도 그렇지 못하다.
불만인 점은, 우리는 Java 로 코딩하며 VM 을 사용하고 있음에도 왜 이런 부분까지 프레임웍에서 챙겨주지 못하냐는 점이다.
이에 대한 불만의 글들이 상당히 많고 메모리 릭 관련 글을 쓴 Android 개발팀의 Romain Guy 가 역관광 당해버린 글타래마저 있으니
참고하자.(http://code.google.com/p/android/issues/detail?id=8488)
위 문제를 해결하기 위해선, Bitmap.recycle() 를 호출해 주어야 하며 ImageView 에 사용한 bitmap 을 recycle 하는 코드는 아래와 같다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 | package net.oomtest; import android.app.Activity; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.Drawable; import android.os.Bundle; import android.util.Log; import android.widget.ImageView; import android.widget.LinearLayout; public class OomtestActivity extends Activity { LinearLayout mainLayout ; ImageView imageView ; @Override protected void onCreate(Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout. main ); mainLayout = (LinearLayout) findViewById(R.id. mainLayout ); imageView = new ImageView( this ); imageView .setImageBitmap(BitmapFactory.decodeFile( "/mnt/sdcard/hwan.documents/images/bigimage/1_portrait.jpg" )); mainLayout .addView( imageView ); } @Override protected void onDestroy() { Log.d( "OOMTEST" , onDestroy" ); recycleBitmap( imageView ); super .onDestroy(); } private static void recycleBitmap(ImageView iv) { Drawable d = iv.getDrawable(); if (d instanceof BitmapDrawable) { Bitmap b = ((BitmapDrawable)d).getBitmap(); b.recycle(); } // 현재로서는 BitmapDrawable 이외의 drawable 들에 대한 직접적인 메모리 해제는 불가능하다. d.setCallback( null ); } } |
코드를 위와 같이 바꾸고 화면을 여러번 돌려보면 OOM Error 없이 화면전환이 잘 되는 것을 확인할 수 있다.
recycle 한줄 호출을 위해 인스턴스 검사, 형변환 까지 해야 하다니 최악이다.
별것 아닌 팁 처럼 보일 것이다.
그러나 왜 이 글을 포스팅하냐면, Android SDK 문서대로 하다가 낭패를 보는 경우가 왕왕 있기 때문이다.
Bitmap.recycle() 에 대한 설명을 보도록 하자.
public void recycle ()
Free the native object associated with this bitmap, and clear the reference to the pixel data. This will not free the pixel data synchronously; it simply allows it to be garbage collected if there are no other references. The bitmap is marked as "dead", meaning it will throw an exception if getPixels() or setPixels() is called, and will draw nothing. This operation cannot be reversed, so it should only be called if you are sure there are no further uses for the bitmap. This is an advanced call, and normally need not be called, since the normal GC process will free up this memory when there are no more references to this bitmap.
영어가 골치아픈 분들을 위해 해석해 보자면 아래와 같다.
public void recycle ()
비트맵에 연관된 네이티브 오브젝트를 정리하고, 픽셀 데이터에 연관된 참조를 끊는다. 이 메소드를 호출한다 하여 픽셀 데이터가 즉시 정리되지는 않는다. 이 비트맵에 대한 다른 참조가 없다면 GC의 대상이 되도록 해 줄 뿐이다. 이 비트맵은 "죽은" 상태가 되어 getPixels() 또는 setPixels() 과 같은 메소드를 호출하면 예외를 발생시킨다. 또한 화면에 아무 것도 그리지 않을 것이다. 이 동작은 되돌릴 수 없으므로 이 비트맵을 더 이상 사용하지 않는다는 확신이 들 경우에만 호출하도록 한다. 이 기능은 고급 기능이며 일반적으로 사용할 필요가 없다. 왜냐하면 일반적인 GC 상황이 되면 비트맵에 대한 참조가 없을 경우 자동으로 GC 대상이 되어 메모리가 해제되기 때문이다.
구라도 이런 구라가 없다.
GC 상황이 되면 메모리가 해제된다고 했는데 구라다.
일반적으로 호출될 필요가 없다고 했는데 안 하면 앱이 죽으므로 구라다.
만약 안드로이드 개발팀이 거짓말을 적어놓은 게 아니라면, 이는 명백한 프레임워크 버그다.
(처음 소스에서 drawable 을 null로 설정했기 때문에 onDestroy 이후에는 네이티브니 뭐니 소리가 나오기 전에 GC가 되어야 한다.)
왜냐하면 위의 코드에서 보듯 imageView 와 bitmap 의 사용 scope 는 onCreate 메소드만으로 한정되어 있기 때문에,
라이프사이클 동안 자동으로 정리되어야 함에도 실상은 그렇지 않기 때문이다.
우리가 반드시 별도의 recycle 과정을 추가로 해 주어야 문제가 사라진다.
또한 메모리 릭 피하기(http://blog.naver.com/nimbusob/147042528) 란 글에서 보면 drawable 을 View 에 주입하면 내부적으로
callback 이 등록되어 어쩌구 저쩌구가 일어나고 이게 Activity context 를 참조하니 마느니 하는 소리를 하는데
개발자가 이런 내부 구현까지 알아야 된다는것 자체가 문제가 있다. 또한 이 문제에 대한 전 세계 개발자들의 불만에 대해
Android team 은 아직까지도 침묵으로 일관하고 있다.
좀더 구체적으로 이야기 하자면, 사실 위 문제는 정확한 GC 타이밍을 잡지 못해 발생한 문제다.
Object lifecycle 과 Activity lifecycle 이 별개이기 때문에 발생하는 문제인 것이다.
심층적으로 분석해 보자면, Bitmap 클래스는 Object#finalize() 메소드를 override 하고 있으며 그 내부에서 native recycle 을 시도하는
구조로 되어 있다. 그런데 이 finalize 메소드는 Dalvik VM의 finalizer thread 가 동작할 때 호출되는데,이 finalizer thread 의 동작 시점이
Activity.onDestroy 와는 완전히 분리되어 있기 때문에 이런 일이 벌어지는 것이다.
정리하자면, Activity가 destroy 되고, Bitmap 의 참조가 0인 상황이며, 새로운 메모리가 필요한 상황을 만들었으나
이 때 Finalizer thread 가 동작하지 않아 참조가 0인 Bitmap 의 메모리가 아직 해제되지 않았고
따라서 OutOfMemoryException 이 발생하는 것이다.
이 문제와 관련하여 더욱 자세한 안드로이드 비트맵과 메모리에 관한 토론글은 (http://stackoverflow.com/questions/1945142/bitmaps-in-android) 에서 확인할 수 있다.
따라서 위와같은 예제 뿐 아니라, xml 레이아웃에서 설정한 Drawble 들도 BitmapDrawable 이라면 명시적으로 recycle 을 해 주는 것이 속편하다. 언제 메모리가 확보될 지 알 수 없는 상황에서 OOM으로 우리 앱이 뻗어버리는 것 보단 나을테니.
이때까지 안드로이드의 프레임웍 자체의 문제를 확인해 보았다. 2.2 이상에서는 해결되었다는 Romain Guy 의 변명글이 있긴 하지만
개발자 커뮤니티에서는 그것조차도 구라라고 하니(테스트환경 2.3.2 에서도 여전히 고쳐지지 않았다) Bitmap 사용에 관해서만큼은
버그이던 뭐던, 시스템을 믿지 말고 구식(old school) 스타일로 우리가 직접 객체 주기를 관리하는 편이 낫다.
또한 이와 비슷한 문제로 View 들의 callback 및 리소스 할당해제에 대한 지저분한 트릭이 있는데
다음 포스트에서 이 문제에 대해 다뤄보겠다.
'IT_Programming > Android_Java' 카테고리의 다른 글
안드로이드 6.0 마시멜로 런타임 권한 적용하기 (0) | 2015.10.21 |
---|---|
Android 디바이스의 고유 번호 (Identifier) 획득 시 고려 해야 할 점 (0) | 2015.10.05 |
[펌] 안드로이드 텍스트 뷰에서 지원하는 HTML 태그들 (0) | 2015.09.10 |
[펌] 안드로이드 오픈지엘(OpenGL ES2.0)의 기본 (0) | 2015.09.08 |
[펌] GLSurfaceView 소개 (0) | 2015.09.08 |