国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統(tǒng) > Android > 正文

Android AIDL和遠(yuǎn)程Service調(diào)用示例代碼

2019-12-12 05:44:15
字體:
供稿:網(wǎng)友

Android:AIDL和遠(yuǎn)程Service調(diào)用

本講的內(nèi)容,理解起來很難,也許你看了很多資料也看不明白,但是用起來缺簡(jiǎn)單的要命。所以我們干脆拿一個(gè)音樂播放器中進(jìn)度條的實(shí)例來說明一下AIDL和Remote Service的價(jià)值和使用方法,你把這個(gè)例子跑一邊,體會(huì)一下就OK了。下面的例子是我

正在準(zhǔn)備的項(xiàng)目實(shí)例中的一部分。

首先說明一下我們面臨的問題,如果看不懂下面的描述請(qǐng)看前面的課程:

第一、我們知道在AndroId中如果需要進(jìn)行音樂播放,最方面的方法就是使用自帶的MediaPlayer對(duì)象,如果我們?cè)贏ctivity中控制MediaPlayer對(duì)象進(jìn)行播放,那么一旦你打開了另外一個(gè)程序譬如瀏覽器,那么歌聲就會(huì)立刻停止,這當(dāng)然不是我們需要的結(jié)果。 我們需要的是在做其他事情的同時(shí)能夠在后臺(tái)聽歌,于是我們就需要把對(duì)MediaPlayer對(duì)象的操作放在后臺(tái)Service中去。

第二、我們已經(jīng)把對(duì)MediaPlayer的操作轉(zhuǎn)移到Service中去了,按照我們以前的做法,我們?cè)贏ctivity中發(fā)送一個(gè)Intent對(duì)象給Service對(duì)象,在Intent中傳送播放啊、暫停啊一類的信息給Service,這樣Service就知道該怎么做了。這一切看起來很美好,可是現(xiàn)在出了一個(gè)新問題,那就是我想在Activity中顯示一個(gè)進(jìn)度條,這個(gè)進(jìn)度條要跟著Service中的MediaPlayer中的歌曲進(jìn)度同步向前走,而且如果我點(diǎn)擊進(jìn)度條中的某一個(gè)位置,還想讓歌曲跳轉(zhuǎn)到新的時(shí)間點(diǎn)繼續(xù)播放,這個(gè),該怎么實(shí)現(xiàn)?

第三、我們需要在Activity中操作Service中的MediaPlayer對(duì)象,就好像這個(gè)對(duì)象是自己的一樣。我們可以采用Android接口定義語言 AIDL(Android Interface Definition Language)技術(shù):

1、把Service中針對(duì)MediaPlayer的操作封裝成一個(gè)接口(.aidl文件)
2、在Service中建個(gè)子類實(shí)現(xiàn)這接口的存根(stub)對(duì)象
3、并在onBind()方法中返回這個(gè)存根對(duì)象。
4、在Activity中使用綁定服務(wù)的方式連接Service,但是不用Intent來傳遞信息,而是在ServiceConnection的onServiceConnected方法里,獲得Service中Stub對(duì)象的客戶端使用代理。我們通過操作Activity中的代理就可以達(dá)到操作Service中的MediaPlayer對(duì)象的目的。這樣我們就可以想用本地對(duì)象一樣操作Service中的對(duì)象了,那么進(jìn)度條一類的需求自然也就迎刃而解。

下面的例子,并不是專門為本講準(zhǔn)備的,所以有些無關(guān)代碼,而且沒加注釋,請(qǐng)見諒(本例完整講解會(huì)放在項(xiàng)目實(shí)訓(xùn)中,正在準(zhǔn)備):

1、新建一個(gè)項(xiàng)目 App_elfPlayer ,啟動(dòng)Activity是個(gè)啟動(dòng)畫面: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個(gè)Activity,1個(gè)Service,還有讀寫外部存儲(chǔ)的權(quán)限聲明3、CoverActivity.java的代碼如下:這是個(gè)全屏的啟動(dòng)畫面,2秒后會(huì)跳轉(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);}

一旦你寫好了這個(gè)IServicePlayer.aidl文件,ADT會(huì)自動(dòng)幫你在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、實(shí)現(xiàn)效果圖:

最后總結(jié)一下,AIDL提供了一種非常簡(jiǎn)單的方式,讓我們可以把一個(gè)進(jìn)程內(nèi)的對(duì)象或方法暴露給另一個(gè)程序使用,就好象另一個(gè)程序也擁有這些功能一樣。

以上就是Android AIDL和遠(yuǎn)程Service 的介紹和簡(jiǎn)單應(yīng)用,后續(xù)繼續(xù)補(bǔ)充相關(guān)知識(shí),謝謝大家的支持!

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 德庆县| 上林县| 大化| 孙吴县| 德州市| 若羌县| 峨眉山市| 同江市| 定日县| 濉溪县| 佳木斯市| 瑞昌市| 杭锦旗| 白城市| 团风县| 通海县| 宜黄县| 鹿邑县| 始兴县| 房产| 伊宁市| 安化县| 徐汇区| 观塘区| 满洲里市| 巍山| 历史| 彩票| 万安县| 大同市| 龙陵县| 商水县| 会同县| 彰化市| 台湾省| 儋州市| 朝阳区| 无极县| 北辰区| 乌审旗| 惠安县|