java 反射

其实原本这篇文章没啥必要,但是最近在 code review 的时候,发现这篇文章中下面这段代码有些疑问,所以就去查了查

      @SuppressLint("NewApi")
    public static boolean isNotificationEnabled(Context context) {
        AppOpsManager appOpsManager = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE);
        ApplicationInfo applicationInfo = context.getApplicationInfo();
        String pkg = context.getApplicationContext().getPackageName();
        int uid = applicationInfo.uid;
        Class appOpsClass;
        try {
            appOpsClass = Class.forName(AppOpsManager.class.getName());
            Method checkOpNoThrowMethod = appOpsClass.getMethod("checkOpNoThrow", Integer.TYPE, Integer.TYPE, String.class);
            Field opPostNotificationValue = appOpsClass.getDeclaredField("OP_POST_NOTIFICATION");
            int value = (Integer) opPostNotificationValue.get(Integer.class);
            return ((Integer) checkOpNoThrowMethod.invoke(appOpsManager, value, uid, pkg) == AppOpsManager.MODE_ALLOWED);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return false;
    }

很明显,这段代码中运用了反射,可是在AppOpsManager 类中的 checkOpNoThrow 方法代码是这样的

    /**
     * Like {@link #checkOp} but instead of throwing a {@link SecurityException} it
     * returns {@link #MODE_ERRORED}.
     * @hide
     */
    public int checkOpNoThrow(int op, int uid, String packageName) {
        try {
            return mService.checkOperation(op, uid, packageName);
        } catch (RemoteException e) {
        }
        return MODE_ERRORED;
    }

方法明明是 public ,自己还愚蠢的写了这么一段代码,看看能不能代替反射的那段代码:

appOpsManager.checkOpNoThrow("OP_POST_NOTIFICATION",uid,pkg)== AppOpsManager.MODE_ALLOWED)

运行,直接crash,报错如下:

java.lang.IllegalArgumentException: Unknown operation string: OP_POST_NOTIFICATION

仔细看了一下,AppOpsManager 类中的 checkOpNoThrow 方法,带有 hide 标签,去一探究竟~

「@hide」标签

类或 API 是否开放,是通过 doc 注释的「@hide」标签来控制的。「@hide」标签表示不对外公开 api,但是系统内部是可以使用该注释标记的接口的。
「@hide」标签注释后的类或者 API 在编译时不对外开放,但是在运行的时候这些类和 API 都是可以访问的。
解决方案:一种是使用反射的方法得到隐藏的 API;一种是使用源码编译时生成的全编译的 jar 包。

java 反射的总结

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

缺点:性能是一个问题,反射相当于一系列解释操作,通知 jvm 要做的事情,性能比直接的 java 代码要慢很多。

关于反射的参考链接:

http://www.cnblogs.com/rollenholt/archive/2011/09/02/2163758.html
http://www.cnblogs.com/whoislcj/p/6038511.html
https://segmentfault.com/a/1190000010162647

关于 hide 标签参考的链接:

https://stackoverflow.com/questions/17035271/what-does-hide-mean-in-the-android-source-code
http://blog.csdn.net/ouyang_peng/article/details/17288253
http://blog.sina.com.cn/s/blog_5da93c8f0101e1yj.html

你可能感兴趣的:(java 反射)