이 글은 『Do it! 깡샘의 안드로이드 앱 프로그래밍 with 코틀린』 교재를 바탕으로 작성되었습니다.
startService()
: 액티비티와 서비스 사이에 관련이 없는 경우bindService()
: 액티비티와 서비스 사이에 관련이 있는 경우class MyBinder : Binder() {
fun funA(arg: Int) {}
fun funB(arg: Int) {
return arg * arg
}
}
override fun onBind(intent: Intent): IBinder? {
return MyBinder()
}
onBind()
가 호출되면 서비스를 실행한 곳에 IBinder 인터페이스를 구현한 객체(예제에서는 MyBinder()
)를 전달함.val connection: ServiceConnection = object : ServiceConnection {
override fun onServiceConnected(name: ComponentName?, service: IBinder?) {
serviceBinder = service as MyService.MyBinder
}
override fun onServiceDisconnected(name: ComponentName?) {
}
}
onBind()
의 리턴값을 전달받음.serviceBinder.funA(10)
class MyService : Service() {
lateinit var messenger: Messenger
internal class IncomingHandler(
context: Context,
private val applicationContext: Context = context.applicationContext
) : Handler(Looper.getMainLooper()) {
override fun handleMessage(msg: Message) {
when (msg.what) {
10 -> Toast.makeText(applicationContext, "${msg.obj}", Toast.LENGTH_SHORT).show()
20 -> Toast.makeText(applicationContext, "${msg.obj}", Toast.LENGTH_SHORT).show()
else -> super.handleMessage(msg)
}
}
}
override fun onBind(intent: Intent): IBinder? {
messenger = Messenger(IncomingHandler(this))
return messenger.binder
}
}
MyService
클래스는 변수 messenger와 내부클래스 IncomingHandler, 함수 handleMessage()
로 이루어져 있다.IncomingHandler
는 Handler(Looper.getMainLooper())
를 상속받았으며, 함수 handleMessage()
를 가진다.handleMessage()
는 외부에서 서비스에 데이터를 전달할 때 자동으로 호출된다.handleMessage()
는 Message를 받아서 message값에 따라 다른 toastMessage를 띄운다.class MainActivity : AppCompatActivity() {
lateinit var messenger: Messenger
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
// 내용 생략
val intent = Intent(this, MyService::class.java)
bindService(intent, connection, Context.BIND_AUTO_CREATE)
}
val connection: ServiceConnection = object: ServiceConnection {
override fun onServiceConnected(name: ComponentName?, service: IBinder?) {
messenger = Messenger(service)
}
override fun onServiceDisconnected(name: ComponentName?) {}
}
}
messenger
에 서비스 컴포넌트의 Handler(HandleMessage)
를 받는다.val msg = Message()
msg.what = 10
msg.obj = "hello"
messenger.send(msg)
Message()
의 what으로 초기화 한 값으로 서비스가 메세지의 내용을 알아볼 수 있도록 한다.Message()
의 obj는 수신자에게 보내는 추상 오브젝트🌍 서비스 컴포넌트를 가진 앱의 매니페스트 파일
<service
android:name=".MyService"
android:exported="true">
<intent-filter>
<action android:name="ACTION_OUTER_SERVICE" />
</intent-filter>
</service>
🌎 서비스 컴포넌트를 이용하는 앱의 매니페스트 파일
<manifest ...생략...>
<queries>
<package android:name="com.example.test_outter" />
</queries>
</manifest>
🌎 서비스를 이용하기 위해 인텐트 객체 생성
val intent = Intent("ACTION_OUTER_SERVICE")
intent.setPackage("com.example.test_outter")
bindService(intent, connection, Context.BIND_AUTO_CREATE)
intent.setPackage()
를 통해 패키지명을 명시한다.🌎 번들 객체를 이용한 A앱의 서비스와 B앱의 액티비티간 데이터 통신
val bundle = Bundle()
bundle.putString("")
package com.example.test_aidl;
interface MyAIDLInterface {
void funA(String data)
int funB();
}
class MyAIDLService : Service() {
override fun onBind(intent: Intent): IBinder {
return object: MyAIDLInterface.Stub() {
override fun funA(data: String?) {
}
override fun funB(): Int {
return 10
}
}
}
}
Stub
은 프로세스 간 통신을 대행해 주는 객체이다.<service
android:name=".MyAIDLService"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="ACTION_AIDL_SERVICE" />
</intent-filter>
</service>
<queries>
<package android:name="com.example.test_outter" />
</queries>
class MainActivity : AppCompatActivity() {
lateinit var aidlService: MyAIDLInterface
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val intent = Intent("ACTION_AIDL_SERVICE")
intent.setPackage("com.example.test_outter")
bindService(intent, connection, Context.BIND_AUTO_CREATE)
}
val connection: ServiceConnection = object : ServiceConnection {
override fun onServiceConnected(name: ComponentName?, service: IBinder?) {
aidlService=MyAIDLService.Stub.asInterface(service)
}
override fun onServiceDisconnected(name: ComponentName?) {
Log.d("kkang", "onServiceDisconnected")
}
}
}
aidlService.funA("hello")