Android AIDL 과 원 격 Service 호출 예제 코드
본 강의 내용 은 이해 하기 가 매우 어렵 습 니 다.아마도 당신 은 많은 자 료 를 보고 도 이해 하지 못 할 것 입 니 다.그러나 사용 하기 에는 간단 한 것 이 부족 합 니 다.그래서 우 리 는 아예 음악 재생 기 에서 진도 바 의 인 스 턴 스 를 가지 고 AIDL 과 Remote Service 의 가치 와 사용 방법 을 설명 하 겠 습 니 다.이 예 를 뛰 어 다 니 며 체험 하면 됩 니 다.아래 의 예 는 나다.
준비 중인 프로젝트 인 스 턴 스 의 일부분 입 니 다.
먼저 우리 가 직면 한 문 제 를 설명 하고 아래 의 설명 을 이해 하지 못 하면 앞의 과정 을 보 세 요.
첫째,우 리 는 Android 에서 음악 재생 이 필요 하 다 는 것 을 알 고 있 습 니 다.가장 좋 은 방법 은 자체 미디어 플레이어 대상 을 사용 하 는 것 입 니 다.만약 에 우리 가 Activity 에서 미디어 플레이어 대상 을 제어 하여 재생 하면 다른 프로그램 을 열 면 브 라 우 저 와 같은 노랫소리 가 바로 멈 추 는 것 입 니 다.이것 은 당연히 우리 가 필요 로 하 는 결과 가 아 닙 니 다.우리 에 게 필요 한 것 은 다른 일 을 하 는 동시에 백 스테이지 에서 노래 를 들 을 수 있 기 때문에 미디어 플레이어 대상 에 대한 조작 을 백 스테이지 서비스 에 두 어야 합 니 다.
둘째,저 희 는 MediaPlayer 에 대한 조작 을 Service 로 옮 겼 습 니 다.예전 의 방법 에 따라 저 희 는 Activity 에서 Intent 대상 을 Service 대상 에 게 보 내 고 Intent 에서 재생,일시 정지 등 정 보 를 Service 에 전송 하면 Service 는 어떻게 해 야 하 는 지 알 수 있 습 니 다.이 모든 것 이 아름 다 워 보이 지만 지금 은 새로운 문제 가 생 겼 습 니 다.그것 은 바로 제 가 Activity 에서 진 도 를 표시 하고 싶 습 니 다.이 진 도 는 Service 중의 MediaPlayer 중의 노래 진 도 를 따라 가 야 합 니 다.그리고 제 가 진도 항목 중의 한 위 치 를 클릭 하면 노래 를 새로운 시간 으로 옮 겨 계속 재생 하고 싶 습 니 다.이것 은 어떻게 실현 해 야 합 니까?
셋째,우 리 는 Activity 에서 Service 중의 MediaPlayer 대상 을 조작 해 야 합 니 다.마치 이 대상 이 자신의 것 과 같 습 니 다.우 리 는 Android 인터페이스 정의 언어 AIDL(Android Interface Definition Language)기술 을 사용 할 수 있 습 니 다.
1.Service 에서 MediaPlayer 에 대한 작업 을 인터페이스 로 밀봉 합 니 다(.aidl 파일)
2.Service 에서 키 클래스 를 만들어 이 인터페이스의 존 근(stub)대상 을 실현 합 니 다.
3.onBind()방법 에서 이 존 근 대상 을 되 돌려 줍 니 다.
4.Activity 에서 바 인 딩 서 비 스 를 사용 하 는 방식 으로 Service 를 연결 하지만 Intent 로 정 보 를 전달 하지 않 고 ServiceConnection 의 onServiceConnected 방법 에서 Service 에서 Stub 대상 의 클 라 이언 트 사용 대 리 를 얻 습 니 다.저 희 는 Activity 의 대 리 를 조작 하면 Service 의 MediaPlayer 대상 을 조작 하 는 목적 을 달성 할 수 있 습 니 다.이렇게 하면 우 리 는 로 컬 대상 처럼 Service 중의 대상 을 조작 하려 고 할 수 있다.그러면 진도 항목 과 같은 수요 도 자 연 스 럽 게 해 결 될 것 이다.
아래 의 예 는 본 강 의 를 위해 특별히 준비 한 것 이 아니 므 로 코드 와 무관 하고 주석 을 달 지 않 았 습 니 다.양해 해 주 십시오.
1.새 항목 AppelfPlayer,Activity 시작 화면:CoverActivity
2.AndroidManifest.xml 의 내용 은 다음 과 같다.
<?xml version="1.0" encoding="utf-8"?>
<manifest package="app.android.elfplayer" xmlns:android="http://schemas.android.com/apk/res/android" android:versioncode="1" android:versionname="1.0">
<uses -sdk="" android:minsdkversion="7">
<uses -permission="" android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses>
<application android:label="@string/app_name" android:icon="@drawable/icon">
<activity android:name=".CoverActivity">
<intent -filter="">
<action android:name="android.intent.action.MAIN">
<category android:name="android.intent.category.LAUNCHER">
</category></action></intent>
</activity>
<activity android:name=".PlayerActivity">
</activity>
<service android:name=".MusicService" android:enabled="true">
</service>
</application>
</uses></manifest>
우 리 는 Activity 2 개,Service 1 개,그리고 외부 에 저 장 된 권한 성명 3,CoverActivity.java 의 코드 를 다음 과 같이 읽 는 것 을 알 게 되 었 습 니 다.이것 은 전체 화면의 시작 화면 입 니 다.2 초 후에 PlayerActivity 로 이동 합 니 다.
package app.android.elfplayer;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.view.Window;
import android.view.WindowManager;
public class CoverActivity extends Activity {
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.cover);
new Handler().postDelayed(new Runnable(){
@Override
public void run() {
Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class);
CoverActivity.this.startActivity(mainIntent);
CoverActivity.this.finish();
}
}, 2000);
}
}
4.PlayerActivity.java 의 코드 는 다음 과 같 습 니 다.
package app.android.elfplayer;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.os.RemoteException;
import android.util.Log;
import android.view.View;
import android.widget.ImageButton;
import android.widget.SeekBar;
import android.widget.SeekBar.OnSeekBarChangeListener;
public class PlayerActivity extends Activity {
public static final int PLAY = 1;
public static final int PAUSE = 2;
ImageButton imageButtonFavorite;
ImageButton imageButtonNext;
ImageButton imageButtonPlay;
ImageButton imageButtonPre;
ImageButton imageButtonRepeat;
SeekBar musicSeekBar;
IServicePlayer iPlayer;
boolean isPlaying = false;
boolean isLoop = false;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.player);
imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite);
imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext);
imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay);
imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre);
imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat);
musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar);
bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);
startService(new Intent(PlayerActivity.this, MusicService.class));
imageButtonPlay.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Log.i("yao", "imageButtonPlay -> onClick");
if (!isPlaying) {
try {
iPlayer.play();
} catch (RemoteException e) {
e.printStackTrace();
}
imageButtonPlay.setBackgroundResource(R.drawable.pause_button);
isPlaying = true;
} else {
try {
iPlayer.pause();
} catch (RemoteException e) {
e.printStackTrace();
}
imageButtonPlay.setBackgroundResource(R.drawable.play_button);
isPlaying = false;
}
}
});
musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() {
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
if (iPlayer != null) {
try {
iPlayer.seekTo(seekBar.getProgress());
} catch (RemoteException e) {
e.printStackTrace();
}
}
}
});
handler.post(updateThread);
}
private ServiceConnection conn = new ServiceConnection() {
public void onServiceConnected(ComponentName className, IBinder service) {
Log.i("yao", "ServiceConnection -> onServiceConnected");
iPlayer = IServicePlayer.Stub.asInterface(service);
}
public void onServiceDisconnected(ComponentName className) {
};
};
Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
};
};
private Runnable updateThread = new Runnable() {
@Override
public void run() {
if (iPlayer != null) {
try {
musicSeekBar.setMax(iPlayer.getDuration());
musicSeekBar.setProgress(iPlayer.getCurrentPosition());
} catch (RemoteException e) {
e.printStackTrace();
}
}
handler.post(updateThread);
}
};
}
5.그 중에서 사용 하 는 IServicePlayer.aidl 은 자바 파일 과 같은 가방 에 넣 고 내용 은 다음 과 같 습 니 다.
package app.android.elfplayer;
interface IServicePlayer{
void play();
void pause();
void stop();
int getDuration();
int getCurrentPosition();
void seekTo(int current);
boolean setLoop(boolean loop);
}
이 IServicePlayer.aidl 파일 을 다 쓰 면 ADT 는 gen 디 렉 터 리 에서 IServicePlayer.java 파일 을 자동 으로 생 성 합 니 다.6.Music Service.java 의 내용 은 다음 과 같다.
package app.android.elfplayer;
import android.app.Service;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;
public class MusicService extends Service {
String tag = "yao";
public static MediaPlayer mPlayer;
public boolean isPause = false;
IServicePlayer.Stub stub = new IServicePlayer.Stub() {
@Override
public void play() throws RemoteException {
mPlayer.start();
}
@Override
public void pause() throws RemoteException {
mPlayer.pause();
}
@Override
public void stop() throws RemoteException {
mPlayer.stop();
}
@Override
public int getDuration() throws RemoteException {
return mPlayer.getDuration();
}
@Override
public int getCurrentPosition() throws RemoteException {
return mPlayer.getCurrentPosition();
}
@Override
public void seekTo(int current) throws RemoteException {
mPlayer.seekTo(current);
}
@Override
public boolean setLoop(boolean loop) throws RemoteException {
return false;
}
};
@Override
public void onCreate() {
Log.i(tag, "MusicService onCreate()");
mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3"));
}
@Override
public IBinder onBind(Intent intent) {
return stub;
}
}
7.실현 효과 도:마지막 으로 AIDL 은 하나의 프로 세 스 내의 대상 이나 방법 을 다른 프로그램 에 노출 시 켜 사용 할 수 있 는 매우 간단 한 방식 을 제공 했다.마치 다른 프로그램 도 이러한 기능 을 가지 고 있 는 것 과 같다.
이상 은 안 드 로 이 드 AIDL 과 원 격 서비스의 소개 와 간단 한 응용 프로그램 입 니 다.추 후 관련 지식 을 계속 보충 하 겠 습 니 다.여러분 의 지지 에 감 사 드 립 니 다!
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Bitrise에서 배포 어플리케이션 설정 테스트하기이 글은 Bitrise 광고 달력의 23일째 글입니다. 자체 또는 당사 등에서 Bitrise 구축 서비스를 사용합니다. 그나저나 며칠 전 Bitrise User Group Meetup #3에서 아래 슬라이드를 발표했...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.