編輯:關於Android編程
Android:AIDL和遠程Service調用
本講的內容,理解起來很難,也許你看了很多資料也看不明白,但是用起來缺簡單的要命。所以我們干脆拿一個音樂播放器中進度條的實例來說明一下AIDL和Remote Service的價值和使用方法,你把這個例子跑一邊,體會一下就OK了。下面的例子是我
正在准備的項目實例中的一部分。
首先說明一下我們面臨的問題,如果看不懂下面的描述請看前面的課程:
第一、我們知道在AndroId中如果需要進行音樂播放,最方面的方法就是使用自帶的MediaPlayer對象,如果我們在Activity中控制MediaPlayer對象進行播放,那麼一旦你打開了另外一個程序譬如浏覽器,那麼歌聲就會立刻停止,這當然不是我們需要的結果。 我們需要的是在做其他事情的同時能夠在後台聽歌,於是我們就需要把對MediaPlayer對象的操作放在後台Service中去。
第二、我們已經把對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、新建一個項目 App_elfPlayer ,啟動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>
我們注意到有2個Activity,1個Service,還有讀寫外部存儲的權限聲明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,放在和Java文件相同的包中,內容如下:
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的內容如下:
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提供了一種非常簡單的方式,讓我們可以把一個進程內的對象或方法暴露給另一個程序使用,就好象另一個程序也擁有這些功能一樣。
以上就是Android AIDL和遠程Service 的介紹和簡單應用,後續繼續補充相關知識,謝謝大家的支持!
一:ViewPager的含義:ViewPager的功能就是可以使視圖滑動,就像Lanucher左右滑動那樣。ViewPager用於實現多頁面的切換效果,該類存在於Goog
本文實例講述了Android編程開發之在Canvas中利用Path繪制基本圖形的方法。分享給大家供大家參考,具體如下:在Android中繪制基本的集合圖形,本程序就是自定
我們在應用中經常看到一些選擇開關狀態的配置文件,做項目的時候用的是android的Switch控件,但是感覺好丑的樣子子個人認為還是自定義的比較好,先上個效果圖:實現過程
在電腦安裝手機游戲軟件時候要用到安卓模擬器,為了工作需要,比如聊天類工具都需要多開,那麼今天講下載使用pkbox安卓模擬器多開的方法。請升級為最新版PXbo