欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2016年08月10日 15:57:59   投稿:lqh  
本文主要介紹Android AIDL和遠(yuǎn)程Service,這里詳細(xì)介紹了相關(guān)知識(shí),并附實(shí)例代碼和實(shí)現(xiàn)效果圖,有興趣的朋友參考下

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

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

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

首先說(shuō)明一下我們面臨的問(wèn)題,如果看不懂下面的描述請(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就知道該怎么做了。這一切看起來(lái)很美好,可是現(xiàn)在出了一個(gè)新問(wèn)題,那就是我想在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接口定義語(yǔ)言 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來(lái)傳遞信息,而是在ServiceConnection的onServiceConnected方法里,獲得Service中Stub對(duì)象的客戶端使用代理。我們通過(guò)操作Activity中的代理就可以達(dá)到操作Service中的MediaPlayer對(duì)象的目的。這樣我們就可以想用本地對(duì)象一樣操作Service中的對(duì)象了,那么進(jìn)度條一類的需求自然也就迎刃而解。

下面的例子,并不是專門為本講準(zhǔn)備的,所以有些無(wú)關(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í),謝謝大家的支持!

相關(guān)文章

  • Android判斷Activity是否在最上層的方法

    Android判斷Activity是否在最上層的方法

    這篇文章主要介紹了Android判斷Activity是否在最上層的方法,涉及Android針對(duì)Activity屬性判斷與操作相關(guān)技巧,代碼非常簡(jiǎn)單易懂,需要的朋友可以參考下
    2016-01-01
  • Activity與Service之間交互并播放歌曲的實(shí)現(xiàn)代碼

    Activity與Service之間交互并播放歌曲的實(shí)現(xiàn)代碼

    以下是對(duì)Activity與Service之間交互并播放歌曲的實(shí)現(xiàn)代碼進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過(guò)來(lái)參考下
    2013-07-07
  • Kotlin?Navigation可視化開發(fā)詳解

    Kotlin?Navigation可視化開發(fā)詳解

    Navigation?是?JetPack?中的一個(gè)組件,用于方便的實(shí)現(xiàn)頁(yè)面的導(dǎo)航,所以抽象出了一個(gè)?destination?的概念,大部分情況一個(gè)?destination?就表示一個(gè)?Fragment,但是它同樣可以指代?Activity、其它的導(dǎo)航圖
    2023-02-02
  • Android高級(jí)開發(fā)之性能優(yōu)化典范

    Android高級(jí)開發(fā)之性能優(yōu)化典范

    本文從電量,視圖,內(nèi)存三個(gè)性能方面的知識(shí)點(diǎn)給大家介紹android高級(jí)開發(fā)之性能優(yōu)化的相關(guān)知識(shí),希望對(duì)大家有所幫助
    2016-05-05
  • 獲取android4.0版本sdcard路徑示例

    獲取android4.0版本sdcard路徑示例

    自從android4.0開始,谷歌為沒有內(nèi)存卡的手機(jī)模擬了一個(gè)SD卡,占用了原來(lái)的SD卡路徑,并為真實(shí)的sd卡掛載到該目錄的子目錄,由于所掛載的目錄并沒有官方規(guī)范,所以命名會(huì)不同,只能通過(guò)搜索,下面是獲取android4.0版本sdcard路徑示例
    2014-03-03
  • Android 中 WebView 的基本用法詳解

    Android 中 WebView 的基本用法詳解

    這篇文章主要介紹了Android 中 WebView 的基本用法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-01-01
  • Android列表動(dòng)圖展示的實(shí)現(xiàn)策略

    Android列表動(dòng)圖展示的實(shí)現(xiàn)策略

    這篇文章主要給大家介紹了關(guān)于Android列表動(dòng)圖展示的實(shí)現(xiàn)策略的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-11-11
  • Android Studio應(yīng)用開發(fā)集成百度語(yǔ)音合成使用方法實(shí)例講解

    Android Studio應(yīng)用開發(fā)集成百度語(yǔ)音合成使用方法實(shí)例講解

    這篇文章主要介紹了Android Studio應(yīng)用開發(fā)集成百度語(yǔ)音合成使用方法實(shí)例講解的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2016-11-11
  • Flutter實(shí)現(xiàn)Text完美封裝

    Flutter實(shí)現(xiàn)Text完美封裝

    本文詳細(xì)講解了Flutter實(shí)現(xiàn)Text完美封裝,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以收藏下,方便下次瀏覽觀看
    2021-11-11
  • Android通過(guò)自定義view實(shí)現(xiàn)刮刮樂效果詳解

    Android通過(guò)自定義view實(shí)現(xiàn)刮刮樂效果詳解

    這篇文章主要介紹了如何在Android中利用自定義的view實(shí)現(xiàn)刮刮樂的效果,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟上小編一起動(dòng)手試一試
    2022-03-03

最新評(píng)論