Android AIDL和远程Service调用示例代码

作者:lqh 时间:2022-07-14 03:06:15 

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、实现效果图:

Android AIDL和远程Service调用示例代码

Android AIDL和远程Service调用示例代码

Android AIDL和远程Service调用示例代码

最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。

标签:Android,AIDL,Service
0
投稿

猜你喜欢

  • MyBatis的嵌套查询解析

    2023-11-26 16:58:46
  • 利用ThreadLocal实现一个上下文管理组件

    2022-04-16 03:42:23
  • Java8函数式接口的基础学习教程

    2023-12-16 12:57:59
  • 使用Feign传递请求头信息(Finchley版本)

    2023-06-07 22:38:07
  • flutter图片组件核心类源码解析

    2023-09-14 16:29:00
  • 简单了解Java字符串(操作)

    2022-04-22 18:37:29
  • 详解如何在Flutter中获取设备标识符

    2022-12-18 12:41:26
  • java使用TimerTask定时器获取指定网络数据

    2022-08-14 10:52:20
  • 第一次编写Java流布局图形界面

    2023-10-13 08:27:11
  • java判断回文数示例分享

    2023-03-20 03:18:22
  • C#计算两个文件的相对目录算法的实例代码

    2022-08-27 10:27:44
  • MyBatis-Plus 集成动态多数据源的实现示例

    2023-06-26 17:56:35
  • SpringBoot去除参数前后空格和XSS过滤

    2023-07-27 02:33:56
  • 解决spring boot启动扫描不到自定义注解的问题

    2023-10-29 14:31:48
  • idea离线使用jrebel的超详细教程

    2022-05-18 10:01:27
  • Java并发编程学习之Unsafe类与LockSupport类源码详析

    2022-10-21 19:07:01
  • java(jdk)环境变量配置(XP、win7、win8)图文教程详解

    2021-11-01 15:06:48
  • C#多线程之任务的用法详解

    2023-08-27 10:51:18
  • Java算法之递归算法计算阶乘

    2021-06-30 14:10:56
  • 使用linq to xml修改app.config示例(linq读取xml)

    2022-11-22 22:01:28
  • asp之家 软件编程 m.aspxhome.com