Android AIDL和遠程Service調(diào)用示例代碼
Android:AIDL和遠程Service調(diào)用
本講的內(nèi)容,理解起來很難,也許你看了很多資料也看不明白,但是用起來缺簡單的要命。所以我們干脆拿一個音樂播放器中進度條的實例來說明一下AIDL和Remote Service的價值和使用方法,你把這個例子跑一邊,體會一下就OK了。下面的例子是我
正在準(zhǔn)備的項目實例中的一部分。
首先說明一下我們面臨的問題,如果看不懂下面的描述請看前面的課程:
第一、我們知道在AndroId中如果需要進行音樂播放,最方面的方法就是使用自帶的MediaPlayer對象,如果我們在Activity中控制MediaPlayer對象進行播放,那么一旦你打開了另外一個程序譬如瀏覽器,那么歌聲就會立刻停止,這當(dāng)然不是我們需要的結(jié)果。 我們需要的是在做其他事情的同時能夠在后臺聽歌,于是我們就需要把對MediaPlayer對象的操作放在后臺Service中去。
第二、我們已經(jīng)把對MediaPlayer的操作轉(zhuǎn)移到Service中去了,按照我們以前的做法,我們在Activity中發(fā)送一個Intent對象給Service對象,在Intent中傳送播放啊、暫停啊一類的信息給Service,這樣Service就知道該怎么做了。這一切看起來很美好,可是現(xiàn)在出了一個新問題,那就是我想在Activity中顯示一個進度條,這個進度條要跟著Service中的MediaPlayer中的歌曲進度同步向前走,而且如果我點擊進度條中的某一個位置,還想讓歌曲跳轉(zhuǎn)到新的時間點繼續(xù)播放,這個,該怎么實現(xiàn)?
第三、我們需要在Activity中操作Service中的MediaPlayer對象,就好像這個對象是自己的一樣。我們可以采用Android接口定義語言 AIDL(Android Interface Definition Language)技術(shù):
1、把Service中針對MediaPlayer的操作封裝成一個接口(.aidl文件)
2、在Service中建個子類實現(xiàn)這接口的存根(stub)對象
3、并在onBind()方法中返回這個存根對象。
4、在Activity中使用綁定服務(wù)的方式連接Service,但是不用Intent來傳遞信息,而是在ServiceConnection的onServiceConnected方法里,獲得Service中Stub對象的客戶端使用代理。我們通過操作Activity中的代理就可以達到操作Service中的MediaPlayer對象的目的。這樣我們就可以想用本地對象一樣操作Service中的對象了,那么進度條一類的需求自然也就迎刃而解。
下面的例子,并不是專門為本講準(zhǔn)備的,所以有些無關(guān)代碼,而且沒加注釋,請見諒(本例完整講解會放在項目實訓(xùn)中,正在準(zhǔn)備):
1、新建一個項目 App_elfPlayer ,啟動Activity是個啟動畫面:CoverActivity
2、AndroidManifest.xml 的內(nèi)容如下:
<?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>
我們注意到有2個Activity,1個Service,還有讀寫外部存儲的權(quán)限聲明3、CoverActivity.java的代碼如下:這是個全屏的啟動畫面,2秒后會跳轉(zhuǎn)到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,放在和Java文件相同的包中,內(nèi)容如下:
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、MusicService.java的內(nèi)容如下:
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、實現(xiàn)效果圖:



最后總結(jié)一下,AIDL提供了一種非常簡單的方式,讓我們可以把一個進程內(nèi)的對象或方法暴露給另一個程序使用,就好象另一個程序也擁有這些功能一樣。
以上就是Android AIDL和遠程Service 的介紹和簡單應(yīng)用,后續(xù)繼續(xù)補充相關(guān)知識,謝謝大家的支持!
相關(guān)文章
Activity與Service之間交互并播放歌曲的實現(xiàn)代碼
以下是對Activity與Service之間交互并播放歌曲的實現(xiàn)代碼進行了詳細的分析介紹,需要的朋友可以過來參考下2013-07-07
Android Studio應(yīng)用開發(fā)集成百度語音合成使用方法實例講解
這篇文章主要介紹了Android Studio應(yīng)用開發(fā)集成百度語音合成使用方法實例講解的相關(guān)資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2016-11-11
Android通過自定義view實現(xiàn)刮刮樂效果詳解
這篇文章主要介紹了如何在Android中利用自定義的view實現(xiàn)刮刮樂的效果,文中的示例代碼講解詳細,感興趣的小伙伴可以跟上小編一起動手試一試2022-03-03

