IT_Programming/Android_Java

[펌] How to quit an application in Android

JJun ™ 2012. 3. 6. 00:52

----------------------------------------------------------------------------------------------

출처: 

http://huewu.blog.me/110120532880

http://blog.naver.com/PostView.nhn?blogId=huewu&logNo=110082677696

http://blog.naver.com/huewu/110089551997

----------------------------------------------------------------------------------------------

 

안드로이드는 멀티태스킹 그리고 어플리케이션 생명주기에 대하여 조금 독특한 철학을 갖고 있습니다. 덕분에, 기존 윈도우나 다른 OS 에서 어플리케이션을 만들었던 개발자들이 안드로이드를 시작할 때, 굉장히 간단해 보이는 일 때문에 고생을 하곤 하는데요. 무엇보다도 '어플리케이션 종료' 에 대하여, 도대체 어떻게 하면 어플리케이션을 종료 시킬 수 있는지 헤매는 경우가 많습니다. (저도 관련해서 여러가지 고생을 했습니다...ㅠㅠ) 난감한 '안드로이 어플리케이션 종료' 에 대하여, 그 동안 삽질한 내용을 기반으로 간단히 5 단계로 정리해 보았습니다.


1. 안드로이드 사전에 '어플리케이션 종료' 는 없습니다.

안드로이드 어플리케이션 생명주기에 관해서 가장 먼저 기억할 점이 있습니다. 

'기본적'으로 안드로이드에는 '어플리케이션 종료' 라는 개념 자체가 없습니다. 안드로이드의 기본정책은 다음과 같습니다.

"한 번 실행된 프로세스는 메모리가 허락하는 한 영원히 죽지 않습니다."

안드로이드 개발팀은 모바일 디바이스 사용자들이 동영상을 보던 중에, 문자 메시지를 확인하고, 전화를 걸다가, 다시 동영상을 시청하는 등으로, 짧은 시간 여러 어플리케이션을 번갈아 가며 사용한다는 점에 주목하고, 최대한 어플리케이션 간의 전환이 빠르고, 사용자가 어플리케이션을 종료한다는 개념 자체를 생각하지 않아도 되도록 프레임워크를 디자인했습니다.

단, 메모리가 부족한 경우 안드로이드 플랫폼은 우선 순위가 낮은 어플리케이션 프로세스를 강제로 종료하여 메모리를 확보고, 이 경우에도 어플리케이션의 마지막 상태 정보를 저장하고 필요한 메모리가 확보되면 이전 상태로 어플리케이션을 복원하는 동작을 수행합니다. 바로 이런 특성 때문에, 강제로 프로세스를 종료해도 어플리케이션이 재시작되는 현상이 일어납니다. 이 주제에 관하여, 개발자 블로그에 올라온 'Multitasking the Android Way' 라는 제목의 포스트를 한 번 읽어보시면 큰 도움이 될 것 같습니다. (번역 되었습니다.)


2. 어플리케이션 종료에는 두 가지 의미가 있습니다.

네. '기본적' 으로 안드로이드 어플리케이션은 종료되지 않습니다. 하지만, 예외없는 규칙은 없는 법. 안드로이드에서도 어플리케이션은 종료될 수 있습니다. 그리고, 어플리케이션 종료는 사실 다음 두 가지의 서로 다른 의미를 뜻할 수 있습니다.

 

  • 유지하고 있던 어플리케이션 상태 정보를 모두 초기화 합니다.
  • 어플리케이션 프로세스를 종료합니다.

 

안드로이드 어플리케이션은 대게 여러 개의 '엑티비티' 와 '서비스' 로 구성됩니다. 엑티비티는 한 화면을 표현하는 기본 단위로, 어플리케이션 진행에 따라 'TASK - 테스크' 라고 불리우는 스택에 하나 씩 차곡 차곡 쌓이게 됩니다. 그리고 이 엑티비티 스택 정보와 해당 어플리케이션에서 동작하고 있는 서비스는 어플리케이션 프로세스에 저장되는 것이 아니라, 안드로이드 시스템 서비스 내부에 간직됩니다.


<실행중인 어플리케이션 정보는 System Service 프로세스 상에서도 관리됩니다.>


 

따라서, 안드로이드 어플리케이션을 종료하는 첫 번째 방법은 바로 시스템 서비스단에 저장된 어플리케이션 상태 정보를 모두 초기화하는 것 입니다. 상태 정보가 모두 초기화되면, 백그라운드로 돌아간 어플리케이션이 재시작할 때, 기존 상태가 복원하는 것이 아니라, 최초로 어플리케이션이 실행된 것 처럼 메인 엑티비티가 실행될 것 입니다. 두 번째 방법은 바로 어플리케이션 프로세스를 종료하는 것 입니다. 프로세스만 종료하면 된다니, 얼핏 생각하면 너무나 간단한 방법이지만, 사실 이 작업이 생각보다 쉽지는 않습니다. 차근 차근 살펴보도록 하겠습니다.


3. 어플리케이션 상태를 초기화하는 방법을 소개 합니다.

안드로이드는 어플리케이션은 다음과 같은 방식으로 상태 정보를 초기화 할 수 있습니다.


  • 사용하는 서비스는 종료 (stopService) 하고 바인딩을 해체(unbindService) 합니다.
  • 등록한 리시버 (registerReceiver) 는 등록을 해지(unRegisterReceiver) 합니다.
  • 사용중인 락(예를 들어 WifiLock) 은 모두 반환(release) 합니다.
  • 엑티비티 테스크를 청소합니다.

 

네. 아쉽게도 특별한 비결은 없습니다. 서비스나 리시버, 락들은 일종의 리소스라고 생각하실 수 있습니다. 사용했던 리소스를 하나 하나 착실히 반환하는 것이 가장 기본이겠지요. 단, 엑티비티 테스크를 비우는 데는 약간의 꽁수가 필요합니다. 기본적으로 안드로이드 시스템은 현재 사용중이 아닌 엑티비티 테스크를 일정 시간 유지합니다. 테스크가 유지되는 동안 사용자가 해당 어플리케이션을 재시작하면 (홈 화면에서 아이콘을 클릭하여) 메인 엑티비티가 실행되는 대신, 테스크가 복원되어 테스크 최상단에 있던 엑티비티가 화면에 표시됩니다. 이를 초기화 하는 데는 다음 두 가지 방법이 사용될 수 있습니다.


 

  • 메니페스트 상에 메인 엑티비티 속성을 "android:clearTaskOnLaunch"값을 "true" 로 설정합니다. 어플리케이션을 종료하고자 할 때, 홈 어플리케이션 엑티비티를 시작하는 방법으로, 어플리케이션을 백그라운드로 전환합니다. 사용자가 어플리케이션을 다시 실행하면, 기존의 테스크 정보가 모두 초기화되고, 항상 메인 엑티비티를 실행합니다. 단, 사용자가 홈키를 누르는 경우에는 어플리케이션 엑티비티 테스크를 유지하고자 한다면, 이 방법을 사용할 수 없습니다. (엑티비티 테스크가 항상 초기화 됨으로)
  • FLAG_ACTIVITY_CLEAR_TOP 플래그를 활용하여, 어플리케이션 메인 엑티비티(루트 엑티비티)를 실행 합니다. 해당 플래그를 사용하면, 이때, 특별한 추가 정보를 포함하여, 실행된 엑티비티가 자기 자신을 바로 종료하도록 구현합니다. 단, 이 경우, 어플리케이션 흐름 상, 메인 엑티비티가 중복해서 실행되지 않도록 주의할 필요가 있습니다. (메인 엑티비티는 루트 엑티비티 역할을 수행함으로 어플리케이션 내에서 메인 엑티비티를 시작할 때는 항상 FLAG_ACTIVITY_CLEAR_TOP | FLAG_ACTIVITY_SINGLE_TOP 플래그를 사용하시는 습관을 갖는 것이 좋습니다.)

 


코드 살펴 볼까요? 우산 첫 번째 방법의 메니페스트 파일은 다음과 같을 것 입니다.

 

<activity android:name=".Main"

android:label="@string/app_name"

android:clearTaskOnLaunch="true">

<intent-filter>

<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />

</intent-filter>

</activity>

현재 실행 중인 어플리케이션은 아래와 같이 홈 어플리케이션을 실행 시키는 방법으로 백그라운드로 전환 시킬 수 있습니다.

 

 

Intent launchHome = new Intent(Intent.ACTION_MAIN);

launchHome.addCategory(Intent.CATEGORY_DEFAULT);

launchHome.addCategory(Intent.CATEGORY_HOME);

startActivity(launchHome);

다음으로 두 번째 방법을 어떤식으로 구현할 수 있는지 살펴보겠습니다. 우선, 메인 엑티비티의 onNewIntent 메서드를 오버라이드 합니다.
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
boolean bFinish = intent.getBooleanExtra("FinishSelf", false);
if(bFinish)
finish();
}
그 후, 어플리케이션을 종료시키는 시점에 (메인 엑티비티가 아니여도 됩니다.) 다음과 같이 CLEAR_TOP 플래그를 활용하고, 미리 정의한 어플리케이션 종료 코드를 EXTRA 로 덧 붙여, 메인 엑티비티를 호출합니다. onResume() 이 호출되기도 전에, finish() 가 호출 되기 때문에, UI 상의 이상한 깜박임 같은 현상 없이 자연스럽게 어플리케이션 테스크가 비워집니다.
Intent clearTop = new Intent(Activity2.this, Activity1.class);
clearTop.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
clearTop.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
clearTop.putExtra("FinishSelf", true);
startActivity(clearTop);
4. Process 종료는 정상적인 해결책은 아닙니다.

 

지금까지 안드로이드에서 기본적으로 제공해주는 방법을 통해 어플리케이션 상태를 초기화 하는 방법에 대하여 살펴보았습니다. 하지만, 한번 시작된 안드로이드 어플리케이션 프로세스는 메모리가 허락하는한 종료되지 않고 계속 살아있습니다. 이 녀석 어떻게 죽일 수 있을까요? 일단, 왜 꼭 프로세스를 종료시켜야 하는지 스스로에게 질문을 던져 봅니다. 어플리케이션이 사용중인 리소스를 정상적으로 잘 정리하였다면, 여러분의 어플리케이션 프로세스는 별다른 문제없이 그저 메모리상에 상주할 뿐 특별한 작업을 수행하진 않을 것 입니다. 더군다나, 사용자가 여러분의 어플리케이션을 재시작하면 어플리케이션이 번개같이 시작될 것이고, 메모리가 부족하면 시스템이 알아서 여러분의 어플리케이션을 프로세스를 종료시켜 줄 것 입니다.


특히 주의할 점이 있습니다. System.exit(0) 이나 Process.killProcess(Process.myPid()) 등의 방식으로 어플리케이션 프로세스를 강제로 종료시키는 것은 결코 좋은 방법이 아닙니다. 여러번 말씀드리지만, 2번 항목의 그림처럼, 안드로이드 어플리케이션의 생명주기와 프로세스의 생명주기는 일치하지 않고, 안드로이드 어플리케이션의 상태정보는 해당 어플리케이션 프로세스 외에 다양한 별도의 시스템 서비스들에 의하여 관리되고 있습니다. (엑티비티 테스크라던가 서비스 정보등등) 만일, 위의 방법을 이용하여 프로세스를 강제로 종료하면, 안드로이드 시스템은 마치 시스템 메모리가 부족하여 해당 프로세스가 종료된 것으로 판단하고, 현재 가용한 시스템 메모리를 확인 한 후, 저장된 인텐트 정보를 활용하여, 해당 어플리케이션을 되살릴 것 입니다. (어플리케이션 종료에 관한 가장 흔한 실수 중에 하나 입니다.)


5. Process 종료는 최후의 해결책입니다.

네. 그럼에도 피치못할 사정으로 프로세스를 종료시켜야 하는 순간이 있습니다. 두 가지 이유가 있을 수 있겠네요.

 

  • 플레쉬 메모리 특성상 여분 메모리가 적으면, 시스템 전체 성능이 저하될 수 있습니다. 따라서, 어플리케이션이 사용하는 메모리가 많고, 어플리케이션 전환이 자주 일어나지 않는 게임 같은 종류의 어플리케인의 경우에는 선제적인 방법으로 프로세스를 종료시키는 편이 좋습니다.
  • 현재 어플리케이션이, 프로세스와 라이프사이클이 일치하는 static 변수들과 무한 루프를 도는 thread 등을 너무 광범위하게 사용하고 있는 경우 (외부 라이브러리 사용등의 이유로) 해당 리소스들을 깔끔하게 종료시키기 위하여 프로세스 종료를 고려해볼 수 있습니다.

 

프로세스를 종료하는 방법은 다음과 같습니다. 우선 프로세스를 종료할 때는, 안드로이드에서 제공하는 ActivityManager.killBackgroundProcess() 메서드를 사용합니다. 이 메서드를 사용하기 위해서 어플리케이션은 KILL_BACKGROUND_PROCESSES 권한을 갖고 있어야 합니다.

<uses-permission android:name="android.permission.KILL_BACKGROUND_PROCESSES"/>


 

ActivityManager am = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);

am.restartPackage(getPackageName());

 

단, 이 메서드는 안드로이드 프로요 이전 버전에서 지원하던 restartPackage 메서드 처럼, 만병 통치약은 아닙니다. 메서드 이름이 암시하는 것 처럼, 오직 어플리케이션이 백그라운드 이하의 중요도를 갖고 있어야지만 정상적으로 동작하지요. 어플리케이션 프로세스의 권한은 다음과 같이 나누어집니다.


 

 

  • IMPORTANCE_EMPTY : 말그대로 빈 깡통. 우선순위가 가장 낮은 프로세스.
  • IMPORTANCE_BACKGROUND : 동작 중인 어플리케이션 컴포넌트가 없는 프로세스.
  • IMPORTANCE_SERVICE : 현재 Service 가 동작하고 있는 프로세스.
  • IMPORTANCE_VISIBLE : 현재 화면상에 보이는 Activity 가 동작하는 프로세스.
  • IMPORTANCE_FOREGROUND : 현재 사용자 U/X 와 직접 연관된 기능을 수행하고 있는 어플리케이션 컴포넌트를 실행 중인 프로세스. 가장 높은 우선 순위를 갖는다.


따라서, 위 메서드를 활용하기 위해서는 현재 실행중인 Foreground 서비스를 정지하고, 현재 테스크를 백그라운드로 돌리는 작업이 선행되어야 합니다. 관련된 방법은 3번 항목에서 한 번 설명드렸으니, 여기서는 엑티비티 매니저가 제공하는 테스크 정보를 기반으로 최대한 자동으로 어플리케이션 프로세스를 종료하는 코드를 소개해보겠습니다.

 

final ActivityManager am = (ActivityManager) getSystemService(ACTIVITY_SERVICE);

//stop running service inside current process.
List<RunningServiceInfo> serviceList = am.getRunningServices(100);
for(RunningServiceInfo service : serviceList){
if( service.pid == Process.myPid()){
Intent stop = new Intent();
stop.setComponent(service.service);
stopService(stop);
}
}

//move current task to background.
Intent launchHome = new Intent(Intent.ACTION_MAIN);
launchHome.addCategory(Intent.CATEGORY_DEFAULT); launchHome.addCategory(Intent.CATEGORY_HOME);
startActivity(launchHome);

//post delay runnable(waiting for home application launching)
new Handler().postDelayed(new Runnable(){
@Override public void run() {
am.killBackgroundProcesses(getPackageName());
}
}, 3000);

코드를 간단히 설명해 볼까요? 기본 원칙은 현재 어플리케이션 프로세스를 백그라운드로 옮긴 후에, 올바른 메서드를 호출하는 것 입니다. 이를 위하여, 현재 동작하는 서비스 중, 어플리케이션 프로세스와 연결된 서비스를 검색하여 강제로 종료시킨 후, 홈 어플리케이션을 시작하여, 현재 어플리케이션 테스크를 백그라운드로 강제 이동시켰습니다. 그리고, 홈 어플리케이션이 정상적으로 시작하기 충분할거라 판단되는 시간 (위 코드에서는 3초)을 기다린 후에, killBackgroundProcesses 메서드를 호출하였습니다. (물론, 어플리케이션 테스크를 내리기 위하여, 앞에서 설명드린 CLEAR_TOP 플래그를 활용하는 방법도 잘 동작합니다.)


결론

안드로이드 어플리케이션의 원칙은 간단합니다. "개발자 여러분! 어플리케이션 종료에 신경 끄세요!!!" 라고 할까요? 그런만큼 안드로이드에서 어플리케이션을 종료하는 일은 생각만큼 간단한 일이 아닙니다. 그럼에도... 우리 개발자들은 언제나 예외가 필요한 법이지요. 하지만, 어플리케이션 종료 코드를 작성하기 전에, 스스로 한번쯤 정말 이런 기능이 필요한지, 그리고 어플리케이션 종료가 정확히 어떤 의미이고, 올바른 사전 작업을 수행했는지 확인할 필요가 있습니다. 그런 순간, 이 포스팅이 도움이 되었으면 좋겠습니다. 그리고, 본 포스팅에서 소개한 방법 외에 어플리케이션을 종료하는 좋은 방법을 아는 분들은 꼭 글 남겨주시면 정말 감사하겠습니다. ^^

 


 

 

안드로이드 어플리케이션 Process 를 정말로 죽이는 방법

 

 안드로이드 상에서 어플리케이션 개발을 진행하다 보면, 어플리케이션 Process 자체를 종료 시키고 싶은 경우가 있습니다. 특히 제 경우에는 여러가지 핸들러나 스레드를 사용하는 경우 Process 를 종료시키고 싶을 때가 많더군요. Activity 를 모두 종료하더라도, Process 가 살아 있으면 메인 UI 스레드에 연결되어 있는 Handler 와 Message 는 쌩쌩히 동작합니다. 더군다나 시간이 오래 걸리는 작업을 수행하기 위해 Thread 를 여럿 생성해 둔 상태에서 해당 Thread 들의 라이프 사이클을 잘 관리하지 않으면 Activity 가 종료 되더라도, 죽지 않고 계속 살아 남기 때문에, 원인을 알 수 없는 오류로 어플리케이션을 죽여 먹곤 했습니다.


그야말로 고생해서 밥 잘 해먹은 다음에, 설겆이 하면서 그릇을 다 깨먹는 형국입니다. 성능상에 약간의 손실을 감수하더라도, 기름때가 잔뜩 묻은 접시마냥 흉칙하게 남아있는 깔끔하게 정리해 버리고자, 몇 가지 방법을 사용해봤습니다. 하지만, 종료 버튼 관련 포스트에서 한번 언급했듯이, 그런 시도들이 썩 성공적이지는 않았습니다.


 

궁하면 통하는 법인가요? 얼마전에 훌륭하게 Process 를 종료시킬 수 있는 방법을 구글링을 통해 발견해서 이야기해 봅니다. 특정 어플리케이션의 Process 를 확실히 죽이는 방법은 다음과 같습니다.
1.메니페스트 파일에 RESTART_PACKAGES 권한을 사용한다고 선언한다.
<uses-permission android:name="android.permission.RESTART_PACKAGES"/>

2.ActivityManager 의 restartPackage API 를 호출한다.
ActivityManager am
= (ActivityManager)getSystemService(ACTIVITY_SERVICE);
am.restartPackage(getPackageName());
이전에는 System.exit() 나 Process.killProcess() 를 이용해서 Process 를 종료하려고 했습니다. 두 가지 방법 모두, Process 를 정상적으로 종료 시키는 듯 보일 때도 있었지만, 대게의 경우 어플리케이션 Process 가 죽었나... 하는 순간에 벌떡 하고 무덤에서 살아 돌아와 저를 우울하게 만들고 했습니다.

어떠한 차이때문에 이런 일이 벌어지는 걸 까요? 안드로이드 Process의 생명 주기에 대해 다시 한번 확인해 봤습니다. 안드로이드 Application Fundamentals 에 명시되어 있는 내용은 다음과 같습니다.

Android may decide to shut down a process at some point, when memory is low and required by other processes that are more immediately serving the user. Application components running in the process are consequently destroyed.
A process is restarted for those components when there's again work for them to do.

System.exit() 이나 Process.killProcess() 를 호출 하면, Process 가 강제로 종료되고, 안드로이드 플랫폼 입장에서는, 예기치않은 이유로 해당 Process 가 종료되었다고 판단하는 것으로 보입니다. 이 때, 해당 Process 가 해야할 일이 남아 있은 경우, 안드로이드 플랫폼 입장에서 이미 무덤에 돌아간 Process 를 다시 불러 일으키게 됩니다. 제가 알고있는 경우는 두 가지 입니다.
  • START_NOT_STICKY 모드가 아닌 Service 가 작동 중인 경우.
  • Activity Task 상에 현재 화면에 보이는 Activity 바로 아래 위치한 Activity 가 종료하고자 하는 Process 의 구성 요소 일 때. (Task 는 Process 가 아니라 안드로이드 시스템에서 관리하고 있음으로, 갑자기 종료된 Activity 바로 아래에 위치한 Activity 를 시스템에서 다시 시작하려고 함으로...)
그럼, restartPackage() 의 경우에는 어째서 Process 가 다시 살아나지 않는 것일까요? 그 답은 API 문서에 잘 나와 있습니다.

public void restartPackage (String packageName)

Have the system perform a force stop of everything associated with the given application package. All processes that share its uid will be killed, all services it has running stopped, all activities removed, etc. In addition, a ACTION_PACKAGE_RESTARTED broadcast will be sent, so that any of its registered alarms can be stopped, notifications removed, etc.


API 설명에 잘 나와 있듯이, restartPackage() 의 경우, 단지 Process 를 종료하는 것이 아니라, 안드로이드 플랫폼에게 특정 어플리케이션 패키지가 종료됨을 알리고, 따라서 이미 실행중인 Service 나 Activity 를 안드로이드 시스템상에서 모두 제거하게 됩니다.

안드로이드 플랫폼은 우리가 알게 모르게 많은 일들을 수행하고 있습니다. 그렇기 때문에, 안드로이드 플랫폼에게 Process 가 죽었다는 사실을 알려준 후에야, 묻어놓은 개발자들은 시체가 무덤에서 다시 돌아오지 않을 것임을 확신하고 마음 편히 잠들 수 있습니다...

 

안드로이드 SDK 가 2.2 Froyo로 업그레이드 된지도 어느정도 시간이 흘렀습니다. 
슬슬 2.2 기반으로 어플리케이션을 작성해야될 순간이 다가오는 셈이지요. 제 경우에도, 얼마전 부터 기존 2.1 기반으로 작성되었던 어플리케이션을 2.2 기반 단말에 올려야 되는 일이 생겼습니다. 처음에는 설마 별 문제 있을까 했는데, 이런, 세상에 공짜 점심은 없는 법 입니다.

 

기존에 유용하게 사용되던, ActivityManager 의 restartPackage() 메서드가 더이상 지원되지 않는 API 가 되어 버렸습니다. 무지막지하게 static 변수와 스레드를 활용하던 기존 코드의 특성(이라고 쓰고 우울함 이라고 읽는다-_-)상, 어플리케이션 상태를 초기화할 필요가 있을 때 종종 사용했습니다만... SDK 문서에 의하면 다음과 같이 변경되었다고 합니다.

public void restartPackage (String packageName)

 

This method is deprecated.
This is now just a wrapper for
killBackgroundProcesses(String); the previous behavior here is no longer available to applications because it allows them to break other applications by removing their alarms, stopping their services, etc.


간단히 말해 restartPackage() 를 대신하여, killBackgroundProcesses() 라는 메서드가 새롭게 추가되었고, 기존 메서드는 간단한 Wrapper 메서드로 변경. 새롭게 추가된 killBackgroundProcesses 를 호출하도록 변경되었다고 합니다.

killBackgroundProcess() 와 프로요 이전 버전의 restartPackage() 메서드의 차이점은 아주 단순합니다. restartPackage() 는 해당 패키지에 속한 모든 프로세스를 모두 종료 시킨 반면, killBackgroundProcess() 는 프로세스의 상태에 따라 중요하지 않는 프로세스만를 선별적으로 종료 시킵니다.

안드로이드 어플리케이션 프로세스의 생명 주기에 관한 내용을 살펴 보면, 안드로이드 프로세스는 5가지 상태로 구분됨을 알 수 있습니다.
  • IMPORTANCE_EMPTY : 말그대로 빈 깡통. 우선순위가 가장 낮다.
  • IMPORTANCE_BACKGROUND : 현재 동작하는 어플리케이션 컴포넌트가 없는 프로세스. 우선 순위는 두 번째로 낮다.
  • IMPORTANCE_SERVICE : 현재 Service 가 동작하고 있는 프로세스.
  • IMPORTANCE_VISIBLE : 현재 화면상에 보이는 Activity 가 동작하는 프로세스. 두 번째로 높은 우선 순이를 갖는다.
  • IMPORTANCE_FOREGROUND : 현재 사용자 U/X 와 직접 연관된 기능을 수행하고 있는 어플리케이션 컴포넌트를 실행 중인 프로세스. 가장 높은 우선 순위를 갖는다.

그리고 테스트 해보니 killBackgroundProcess() 메서드는 IMPORTANCE_SERVICE 를 포함하여, 그보다 낮은 중요도를 갖는 프로세스만을 종료시키는 것으로 파악됩니다. 즉, 기존과는 다르게 멀쩡하게 돌고있는 Activity 메인 스레드 내에서 대충 편한 시점에 restartPackage() 를 호출 해서는 현재 동작하는 프로세스를 죽일 수 없다는 말 입니다.

즉, 간단한 예로 안드로이드 프로요 2.2 에서는 다음과 같은 코드는 어플리케이션 Process 를 종료시키지 않습니다. (onCreate() 가 호출된 시점은 Activity 가 멀쩡히 작동 하고 있는 상태이기 때문입니다.)

@Override
public void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.main);
 ActivityManager am = (ActivityManager) getSystemService
(Context.ACTIVITY_SERVICE); am.restartPackage(getPackageName()); }
대신, onDestroy() 같은 시점에 호출하면, 정상적으로 프로세스를 종료시키게 되지요. 물론 해당 프로세스에 현재 동작하는 Activity 외에 다른 별도의 어플리케이션 컴포넌트가 없는 경우에만 해당되는 이야기 입니다만.
뭐, 호출 시점을 좀 조심하면 해결되는 문제이면 좋겠지만, 한가지 애매한 경우가 있습니다. 바로 해당 어플리케이션이 Service 를 사용 하고 있는 경우입니다. 약간 버그성 갖긴 합니다만, Service 가 동작하고 있는 상황에서, Activity 의 onDestroyl시점에 killBackgroundProcesses() 메서드를 호출하는 경우, 안드로이드 시스템은 우선 프로세스를 종료하긴 합니다만, 곧 이어서 동작 중인 Service 가 비정상적으로 종료되었다고 판단하고, Service 시작 시 설정된 Start Command 에 따라, 기존 프로세스를 다시 되살리는 등의 일을 수행합니다. (차라리 죽이질 말던가...) 그렇다고, stopService() 메서드를 호출 한 후에, killBackgroundProcesses() 를 호출한다 하면 해결될 문제도 아닌 것이, stopService 는 IPC 호출이 일어나는 넌블럭킹 함수인 만큼, 미묘한 타이밍 문제를 일으킬 수 있더군요.

따라서, Service 를 사용하고 있는 어플리케이션은 killBackgroundProcesses() 메서드의 호출 시점에 좀 더 주의를 기울여야 합니다. 사용 중인 Service 를 확실하게 종료시키고, 프로세스의 중요도가 IMPORTANCE_BACKGROUND 보다 낮은 것을 확인 한 후에 killBackgroundProcesses() 를 호출하는 것이 안전합니다.

다음의 코드는, 안드로이드 SDK 버전과 상관없이 (1.6 이상이면...) 동작하는, 안드로이드 어플리케이션 Process 를 얼추 죽이는 나름의 방법에 관한 예제입니다.

public void requestKillProcess(final Context context){
 // #1. first check api level.
 int sdkVersion = Integer.parseInt(Build.VERSION.SDK);
 if (sdkVersion < 8){
  // #2. if we can use restartPackage method, just use it.
  ActivityManager am = (ActivityManager) context.getSystemService(
Context.ACTIVITY_SERVICE); am.restartPackage(getPackageName()); }else{ // #3. else, we should use killBackgroundProcesses method. new Thread(new Runnable() { @Override public void run() { ActivityManager am = (ActivityManager) context.getSystemService(
Context.ACTIVITY_SERVICE); String name = getApplicationInfo().processName; RunningServiceInfo si; // pooling the current application process importance
// information. while(true){ List<RunningAppProcessInfo> list = am.getRunningAppProcesses(); for(RunningAppProcessInfo i : list){ if(i.processName.equals(name) == true){ // #4. kill the process, // only if current application importance is less than
// IMPORTANCE_BACKGROUND if(i.importance >= RunningAppProcessInfo.
IMPORTANCE_BACKGROUND) // simple wrapper of killBackgrounProcess
       am.restartPackage(getPackageName());
  else Thread.yield(); break; } } } } }, "Process Killer").start(); } }

SDK 버전과 관계없이 프로세스를 종료할 때 사용할 수 있는 requestKillProcess() 라는 메서드를 구현하였습니다. 프로요 이전 버전인 경우, 기존에 사용하던 방법이 그대로 별다른 생각없이 restartPackage() 메서드를 호출 하도록 구현되어 있습니다.

프로요 이 후 버전인 경우가 조금 골치 아픈데, 어느 시점에 해당 어플리케이션 프로세스의 우선 순위가 변경될지 알아내는 방법을 잘 몰라서, 주기적으로 프로세스 상태 정보를 폴링하도록 구현되었습니다. 스레드를 하나 생성한 후, 프로세스의 상태를
반복 체크하다가, 프로세스 중요도가
IMPORTANCE_BACKGROUND 보다 낮아지는
순간,
restartPackage() 를 호출 하도록 되어있습니다.

어째, 구현 내용이 좀 꺼림직 하긴 합니다만, 어플리케이션 내에서 사용하는 어플리케이션 컴포넌트 (Service / Activity / BroadcastReceiver ) 관리에만 신경을 써 준다면 (시작한 Service 는 잊지 않고 정지 시켜주는 등), 기존 restartPackage() 와 같이
어플리케이션 어느 시점에 호출에도 얼추 비슷하게 동작하는 것은 확인 하였습니다.

(스레드가 계속 도니까;;;;)

아무리 생각해봐도 더 좋은 방법이 있을거 같긴 합니다만;;;
딱히 다른 방법이 떠오르지 않네요.
더 좋은 방법을 알고계신 분은
꼭 코멘트 해주시길 바랍니다;;;

 

 


 

더보기

[1]

i.importance = RunningAppProcessInfo.IMPORTANCE_EMPTY;
am.killBackgroundProcesses(i.processName);
이렇게 직접적으로 Empty 를 넣어주면 kill 작동이 좀더 향상 되네요

 

[2]

moveTaskToBack(true);
finish();

android.os.Process.killProcess(android.os.Process.myPid());

ActivityManager am = (ActivityManager)getSystemService(ACTIVITY_SERVICE); am.restartPackage(getPackageName());

프로요 종료시 이 소스를 함 참조해 주세요.


[3] (2017.07.20)

if (Build.VERSION.SDK_INT >= 21)

    finishAndRemoveTask();

else

    finish();

System.exit(0);


* 출처: https://link2me.tistory.com/1331 [소소한 일상 및 업무TIP 다루기]