java 反射

作者: badmask | 来源:发表于2017-09-26 19:47 被阅读0次

    其实原本这篇文章没啥必要,但是最近在 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 反射

        本文链接:https://www.haomeiwen.com/subject/ziheextx.html