android 使用反射获取MediaPlayer的Invoke方法

原创 2016年08月29日 19:25:56

最近有需求需要使用MediaPlayer的invoke接口去实现某些功能, 但是invoke接口是隐藏的, 没有在sdk中开放出来. 所以使用反射的方法来获取invoke接口, 但在实现的过程中出现一些问题, 在这里记录一下.

1.使用反射的方式获取隐藏的接口

    if (mMediaPlayer != null) {
        Parcel request = Parcel.obtain();
        Parcel reply = Parcel.obtain();
        try {
            request.writeInt(200);

            Class<?> cls = mMediaPlayer.getClass();
            Method method = cls.getDeclaredMethod("invoke", Parcel.class, Parcel.class);
            method.setAccessible(true); //如果隐藏接口是public的, 这句可以不要
            method.invoke(mMediaPlayer, request, reply); 

            int result = reply.readInt();
            if (0 == result) {
              return false;
          } else if (1 == result) {
              return true;
          }
        }  catch (Exception e) {
                e.getCause().printStackTrace();
        } finally {
            request.recycle();
            reply.recycle();
        }
    }

反射的调用步骤:
(1) 获取相关类的class
Class cls = mMediaPlayer.getClass();
或者通过包名获取:
Class cls = Class.forName (“android.media.MediaPlayer”);

(2) 通过方法名获得方法接口, 如果方法有参数, 需要将参数的class传入
Method method = cls.getDeclaredMethod(“invoke”, Parcel.class, Parcel.class);

(3) 通过Method的invoke接口来实现方法的调用, 这时候需要传参, 将参数传入:
method.invoke(mMediaPlayer, request, reply);

2.出现的问题
当通过这种方式调用后会发现无效, 并且会打印下面的error信息:

01-01 08:31:42.270 W/System.err( 1475): java.lang.reflect.InvocationTargetException
01-01 08:31:42.270 W/System.err( 1475):  at java.lang.reflect.Method.invokeNative(Native Method)
01-01 08:31:42.270 W/System.err( 1475):  at java.lang.reflect.Method.invoke(Method.java:511)

这个问题搞了好久, 反射调用的方法应该没什么问题, 错误应该是其他的, 仔细看log发现有下列信息:

01-01 08:31:42.269 E/Parcel  ( 3212): Reading a NULL string not supported here.

我没有传入任何string下去, 却报出这个error, 应该是需要再出入一个string下去.

这个时候查看MediaPlayer的源码, 看到selectOrDeselectTrack等方法内部也是调用的invoke接口:

    private void selectOrDeselectTrack(int index, boolean select)
            throws IllegalStateException {
        Parcel request = Parcel.obtain();
        Parcel reply = Parcel.obtain();
        try {
            request.writeInterfaceToken(IMEDIA_PLAYER);
            request.writeInt(select ? INVOKE_ID_SELECT_TRACK
                    : INVOKE_ID_DESELECT_TRACK);
            request.writeInt(index);
            invoke(request, reply);
        } finally {
            request.recycle();
            reply.recycle();
        }
    }

这里面与我写的调用invoke接口多了一句:

request.writeInterfaceToken(IMEDIA_PLAYER);

IMEDIA_PLAYER的值为:

  private final static String IMEDIA_PLAYER = "android.media.IMediaPlayer";

当我将这句加上测试后, 果然没问题了, 调用成功. 这句代码的意思是标识远程服务的名称. 不然不知道去启动哪个服务来操作.

其实MediaPlayer中有提供方法来获取Parcel对象, 但是此方法也是隐藏的:

    public Parcel newRequest() {
        Parcel parcel = Parcel.obtain();
        parcel.writeInterfaceToken(IMEDIA_PLAYER);
        return parcel;
    }

这个方法的内部也是调用了writeInterfaceToken接口.

3.其他
如果可以查看源码, 其实每个方法的使用源码中都有对应的test例子, 例如现在说的invoke方法, 在源码中有 MediaPlayerInvokeTest.java, 这里面介绍了如何使用invoke接口, 下面是这个类的内容:

    // Tests for the invoke method in the MediaPlayer.
    public class MediaPlayerInvokeTest extends
            ActivityInstrumentationTestCase2<MediaFrameworkTest> {
        private static final String TAG = "MediaPlayerInvokeTest";
        private MediaPlayer mPlayer;
        private Random rnd;

        public MediaPlayerInvokeTest() {
            super("com.android.mediaframeworktest", MediaFrameworkTest.class);
            rnd = new Random(Calendar.getInstance().getTimeInMillis());
        }

        @Override
        protected void setUp() throws Exception {
            super.setUp();
            mPlayer = new MediaPlayer();
        }

        @Override
        protected void tearDown() throws Exception {
            mPlayer.release();
            super.tearDown();
        }

        // Generate a random number, sends it to the ping test player.
        @Suppress
        @MediumTest
        public void testPing() throws Exception {
            mPlayer.setDataSource("test:invoke_mock_media_player.so?url=ping");

            Parcel request = mPlayer.newRequest();
            Parcel reply = Parcel.obtain();

            int val = rnd.nextInt();
            request.writeInt(val);
            mPlayer.invoke(request, reply);
            assertEquals(val, reply.readInt());
        }
    }

这次的收获就是要学会善于查看android源码.

版权声明:本文为博主原创文章,未经博主允许不得转载。

安卓开发之屏蔽按键抖动,连击,长按事件

安卓开发之屏蔽按键抖动,连击,长按事件

从零开始学android:Android事件处理—长按事件

长按事件 在Android中提供了长按事件的处理操作,所谓的长按事件就好象读者经常使用某一个手机,长按某一个组件2秒之后才会触发这一操作,而不像普通的单击事件那样,每次单击都会执行一次,长按事件使用V...

Android MediaPlayer使用方法简单介绍

  • 2012年06月19日 22:34
  • 13.59MB
  • 下载

android MediaPlayer的三种使用方法

  • 2012年02月14日 13:48
  • 68KB
  • 下载

java中通过反射获取方法并且调用(getMethod和invoke深入)实践

为了支持业务的快速变更,往往采用可配置的方式,将业务逻辑的处理部分配置在数据库中或者XMl文件里。配置什么,如何配置才更灵活,That's a problem. 以数据库配置为例(xml相同),...
  • n8765
  • n8765
  • 2016年04月28日 10:15
  • 323

java中通过反射获取方法并且调用(getMethod和invoke深入)实践

为了支持业务的快速变更,往往采用可配置的方式,将业务逻辑的处理部分配置在数据库中或者XMl文件里。配置什么,如何配置才更灵活,That's a problem. 以数据库配置为例(xml相同),在数据...
  • wawmg
  • wawmg
  • 2013年12月17日 00:43
  • 30755

JavaSE_Java反射机制详解(Field,Method) 及 Method.invoke使用方法

JAVA反射机制 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称...

java反射Method中的如何调用任意方法,即invoke()的使用

今天碰到了一个关于java反射中的Method中invoke(Object,Object...args);的使用问题,这就涉及到了反射中如何去利用这个方法去调用一个类中任意方法的问题,包括类中的静态和...

【Android】MediaPlayer使用方法简单介绍

获得MediaPlayer实例 可以使用直接new的方式: MediaPlayer mp = new MediaPlayer(); 也可以使用create的方式,如: MediaPlayer mp =...

深入Android MediaPlayer的使用方法详解

1)如何获得MediaPlayer实例: 可以使用直接new的方式: MediaPlayer mp = new MediaPlayer(); 也可以使用create的方式,如: MediaPl...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:android 使用反射获取MediaPlayer的Invoke方法
举报原因:
原因补充:

(最多只允许输入30个字)