Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 747|回复: 0

第二十二讲:AIDL和远程Service调用 -  Android学习

[复制链接]

该用户从未签到

发表于 2011-10-25 09:02:11 | 显示全部楼层 |阅读模式
本讲内容:AIDL和远程Service调用
本讲源代码:  App_elfPlayer.zip (1.57 MB, 下载次数: 30)
本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下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提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素,好了,本讲就到这里。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

GMT+8, 2025-1-11 14:30 , Processed in 0.308276 second(s), 36 queries .

Powered by Discuz! X3.4

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表