关于Class的一个坑

今天遇到一个关于Class类的坑,分享给大家。
有一个需求是,通过intent启动一个Activity来加载某插件,要在intent中传递插件的代理类(Class)到Activity中,由于intent里面只能传递基本类型或是可序列化的类型,故而将需要传递的所有参数全部放入json字符串中,在Activity中对json进行解析,得到其中代理类Class的名字,再利用Class类的方法Class.forName()获取该类名所对应的代理类。
示意代码如下:
发送方:

    private void startInstallActivity(Context context) {
        Intent intent = new Intent(context, MyInstallActivity);
        JSONObject jsonObject = new JSONObject();
        try {
            jsonObject.put("mProxyActivityClass",Activity_A.class);
            // 加入其它需要传递的参数……
        } catch (JSONException e) {
            e.printStackTrace();
        }
        intent.putExtra("jsonObject", jsonObject.toString());
        context.startActivity(intent);
    }

在接收方MyInstallActivity中,收到intent启动后,解析:

    private Class parseIntent(Intent intent) {
        Class proxyClass = null;
        String jsonObjStr = intent.getStringExtra("jsonObject");
        try {
            JSONObject jsonObject = new JSONObject(jsonObjStr);
            String theClassStr = jsonObject.optString("mProxyActivityClass");
            Log.i(TAG, theClassStr);
            //解析其它参数……
            try {
                proxyClass = Class.forName(theClassStr);
            } catch (ClassNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        } catch (JSONException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return proxyClass;
    }

预期的情况是,在intent中传入Activity_A的类名,然后在接收方MyInstallActivity中,通过解析jsonString来得到Activity_A的类名,然后通过类加载器得到Activity_A类(即是需要用到的启动插件的代理类)。
然而,在实际运行过程中得到的结果并非与所预期的一样,jsonString中取得的类名无法解析出对应的类!
通过打印log,发现theClassStr字符串为“class xxx.Activity_A”,即是有一个前缀class加空格,再跟着是包名加类名,用Class.forName()无法得到对应的类!
……
再查看Class的相关API,分析一下,传入intent的类名实际是class.toString(),也就是类的默认toString方法,那么就看一下toString方法的源码:

@Override
    public String toString() {
        if (isPrimitive()) {
            return getSimpleName();
        } else {
            return (isInterface() ? "interface " : "class ") + getName();
        }
    }

它的含义是,若该类是基本类型(预定义的基本类型如int,boolean,byte,char,short,long,float,double等),则返回其类名,否则返回一个前缀加上完整类名,前缀根据是接口还是类,分为”interface ” 和”class “。至此,真相大白!
通过toString得到的类名并不是我们预期的完整类名,但是通过看这里的源码可以推测出getName()方法可能是我们预期的完整类名。继续看一下其源码:

    /**
     * Returns the name of the class represented by this {@code Class}. For a
     * description of the format which is used, see the class definition of
     * {@link Class}.
     *
     * @return the name of the class represented by this {@code Class}.
     */
    public String getName() {
        String result = name;
        return (result == null) ? (name = getNameNative()) : result;
    }

    private native String getNameNative();

通过源码,看到getNameNative()是native方法,这里并不能得到确切的解释其返回结果为类的包名+类名,那就通过自己写一个demo实例来具体验证一下吧:

    private void test() {
        String name = "" + MainActivity.class;
        Log.w(TAG, "MainActivity.class = "+name);
        nameOfClassForName(name);
        name = "" + MainActivity.this.getClass();
        Log.w(TAG, "MainActivity.this.getClass() = "+name);
        nameOfClassForName(name);
        name = "" + MainActivity.this.getClass().getName();
        Log.w(TAG, "MainActivity.this.getClass().getName() = "+name);
        nameOfClassForName(name);
        name = "" + MainActivity.class.getName();
        Log.w(TAG, "MainActivity.class.getName() = "+name);
        nameOfClassForName(name);
        name = "" + MainActivity.class.getSimpleName();
        Log.w(TAG, "MainActivity.class.getSimpleName() = "+name);
        nameOfClassForName(name);
    }

    private void nameOfClassForName(String name) {
        try {
            Class<?> theClass = Class.forName(name);
            Log.w(TAG, "Class.forName = " + theClass);
        } catch (ClassNotFoundException e) {
            Log.e(TAG, "Class.forName is null!");
        }
    }

运行一下,看看log吧:
log
仔细分析,发现通过this.getClass().getName()和class.getName()最后得到的结果相同,都是完整的包名+类名。

Get!找到答案了,最终要用class.getName方法!
只要修改原先代码的一行就ok:

……
jsonObject.put("mProxyActivityClass",Activity_A.class.getName());
……

此坑,记下了,各位同学下次遇到这种奇葩问题,可以通过分析源码和实际demo验证来解决!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于memset函数来说,它是用来初始化一块内存的函数,并将这块内存中的所有字节都设置为指定的值。因此,无法直接使用memset函数来初始化一个pair类型的变量。因为pair是一个包含两个不同类型的元素的结构体,无法简单地将整个pair对象的内存设置为指定的值。如果你想要初始化一个pair对象,可以使用make_pair函数来创建一个pair对象,并为其提供初始值。例如,使用make_pair<string, int>(string(str), int(pos))可以创建一个包含一个string类型和一个int类型元素的pair对象。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [stl中的对组pair、C++常用库函数](https://blog.csdn.net/weixin_72765295/article/details/128400410)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [编程时踩过的(主要是c++)](https://blog.csdn.net/hyk_1996/article/details/96304253)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [leetcode2sumc-LeetCode_Solutions:剑指offer、Leetcode题解c++版本](https://download.csdn.net/download/weixin_38680475/20046128)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值