获取Java匿名内部类持有的外部类对象

熟悉Java的应该都知道,Java匿名内部类会隐式持有一个外部类对象。所以在匿名内部类里可以调用外部类各个方法。

public interface Callback {
    void callback(String s);
}
public class Main {

    private String mName;

    public Main(String name) {
        mName = name;
    }

    public static void main(String[] args) {
        Main main = new Main("Main");
        main.test();
    }

    private void test() {
        Callback callback = new Callback() {

            @Override
            public void callback(String s) {
                System.out.println(getName());
            }
        };
        callback.callback("");
    }

    private String getName() {
        return mName;
    }
}

这段代码很简单,new出来的Callback内部可以调用外部Main的getName方法。持有的外部对象是Java自动完成的,不需要自己手动处理。
最近遇到一个特殊的需求,可以拿到一个匿名内部类,然后要获取他持有的外部类对象。需求有点怪,抛开奇怪本身,作为技术层面问题,思考下如何拿到外部类对象?
既然匿名内部类持有了外部类对象,怎么持有?猜测匿名内部类肯定有个隐藏的属性。那先把匿名内部类持有的所有属性打印出来。把test方法改造成下面代码:

    private void test() {
        Callback callback = new Callback() {
            @Override
            public void callback(String s) {
                System.out.println(getName());
            }
        };
        callback.callback("");
        Class clazz = callback.getClass();
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            System.out.println(field.getName());
            try {
                System.out.println(field.get(callback));
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }
    }

控制台输出了:

Main
this$0
com.aesean.clazz.Main@330bedb4

Main是Callback输出的,很显然,这里的this$0就是外部类对象的属性名。这下简单了,如果要获取一个匿名内部类持有的外部类对象,我们反射属性this$0不就可以了。

       try {
            Field this$0Field = clazz.getDeclaredField("this$0");
            System.out.println("反射this$0 = "+this$0Field.get(callback));
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }

test方法最后加上上面代码,运行下控制台输出:

Main
this$0
com.aesean.clazz.Main@330bedb4
反射this$0 = com.aesean.clazz.Main@330bedb4

很显然我们反射拿到了外部类对象。但是这里有个问题,这里this$0并不是特殊名字,如果匿名内部类本来就有个属性名字叫this$0怎么办?

    private void test() {
        Callback callback = new Callback() {
            String this$0 = "callback_this0";

            @Override
            public void callback(String s) {
                System.out.println(getName());
            }
        };
        callback.callback("");
        Class clazz = callback.getClass();
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            System.out.println(field.getName());
            try {
                System.out.println(field.get(callback));
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }

        try {
            Field this$0Field = clazz.getDeclaredField("this$0");
            System.out.println("反射this$0 = "+this$0Field.get(callback));
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }

代码改成上面的样子,再运行看下输出。

Main
this$0
callback_this0
this$0$
com.aesean.clazz.Main@330bedb4
反射this$0 = callback_this0

问题出现了,反射this$0拿到的是我们自己新建的String。而Fields那里多了一个新对象this$0$,很显然它是我们的外部类对象。那是不是意味着外部类对象会优先以this$0属性名存在,如果相同名字存在就多加一个$符号用来区分?

            String this$0 = "callback_this0";
            String this$0$ = "callback_this0$";
            String this$0$$ = "callback_this0$$";

多加几个属性,控制台输出:

Main
this$0
callback_this0
this$0$
callback_this0$
this$0$$
callback_this0$$
this$0$$$
com.aesean.clazz.Main@330bedb4
反射this$0 = callback_this0

似乎还真是这样,那我们是不是可以通过拿最多$符号的对象当成是外部类对象呢?也不行。

    private void test() {
        Callback callback = new Callback() {
            String this$0 = "callback_this0";
//            String this$0$ = "callback_this0$";
            String this$0$$ = "callback_this0$$";

            @Override
            public void callback(String s) {
                System.out.println(getName());
            }
        };
        callback.callback("");
        Class clazz = callback.getClass();
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            System.out.println(field.getName());
            try {
                System.out.println(field.get(callback));
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }

        try {
            Field this$0Field = clazz.getDeclaredField("this$0");
            System.out.println("反射this$0 = "+this$0Field.get(callback));
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }

注释掉属性this$0$,看下输出:

Main
this$0
callback_this0
this$0$$
callback_this0$$
this$0$
com.aesean.clazz.Main@330bedb4
反射this$0 = callback_this0

很显然这时候this$0$变成了外部类对象。单纯通过名字是不可能完全正确判断的,那怎么办?
熟悉反射的应该都知道,Class或者Field都有一个getModifiers();方法。modifier是Class或者Field的修饰符,记录了Class和Field的一些特性。另外还有一个Modifier类,提供了一些静态方法帮助判断Class和Field的一些特性。
我们再把test方法改造下。

    private void test() {
        Callback callback = new Callback() {
            String this$0 = "callback_this0";
            //            String this$0$ = "callback_this0$";
            String this$0$$ = "callback_this0$$";

            @Override
            public void callback(String s) {
                System.out.println(getName());
            }
        };
        callback.callback("");
        Class clazz = callback.getClass();
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            try {
                System.out.println(field.getName() + " + " + field.get(callback) + " + " + Integer.toHexString(field.getModifiers()));
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }

        try {
            Field this$0Field = clazz.getDeclaredField("this$0");
            System.out.println("反射this$0 = " + this$0Field.get(callback));
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }

控制台输出

Main
this$0 + callback_this0 + 0
this$0$$ + callback_this0$$ + 0
this$0$ + com.aesean.clazz.Main@330bedb4 + 1010
反射this$0 = callback_this0

匿名内部类的modifier数值十六进制表示是1010,我们去Modifier类去对照下。

    public static final int FINAL = 0x00000010;
    ......
    static final int SYNTHETIC = 0x00001000;

很显然这里的1010表示FINAL+SYNTHETIC. SYNTHETIC中文意思就是合成的,其实就是表示Field或者Class是编译器自动生成的意思。那这下简单了。我们去反射this$0对象,然后判断modifier是否为0x1010,如果是可以认为它就是外部类对象,如果不是那我们继续反射this$0$,以此类推。整理成代码就是下面这样:

public class Main {

    private String mName;

    public Main2(String name) {
        mName = name;
    }

    public static void main(String[] args) {
        Main2 main = new Main2("Main");
        main.test();
    }

    private void test() {
        Callback callback = new Callback() {
            String this$0 = "callback_this0";
            // String this$0$ = "callback_this0$";
            String this$0$$ = "callback_this0$$";

            @Override
            public void callback(String s) {
                System.out.println(getName());
            }
        };
        try {
            System.out.println("获取外部类对象:" + getExternalClass(callback));
        } catch (NoSuchFieldException e) {
            throw new RuntimeException(callback.getClass().getName() + "不是一个匿名内部类,或者该匿名内部类没有持有外部类对象。", e);
        }
    }

    private String getName() {
        return mName;
    }

    private static final int SYNTHETIC = 0x00001000;
    private static final int FINAL = 0x00000010;
    private static final int SYNTHETIC_AND_FINAL = SYNTHETIC | FINAL;

    private static boolean checkModifier(int mod) {
        return (mod & SYNTHETIC_AND_FINAL) == SYNTHETIC_AND_FINAL;
    }

    public static Object getExternalClass(Object target) throws NoSuchFieldException {
        return getField(target, null, null);
    }

    private static Object getField(Object target, String name, Class classCache) throws NoSuchFieldException {
        if (classCache == null) {
            classCache = target.getClass();
        }
        if (name == null || name.isEmpty()) {
            name = "this$0";
        }
        Field field = classCache.getDeclaredField(name);
        field.setAccessible(true);
        if (checkModifier(field.getModifiers())) {
            try {
                return field.get(target);
            } catch (IllegalAccessException e) {
                throw new RuntimeException(e);
            }
        }
        return getField(target, name + "$", classCache);
    }
}

运行输出:

获取外部类对象:com.aesean.clazz.Main@330bedb4

最终结果没有受到类似名字属性的干扰成功拿到了外部类对象。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,604评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,642评论 18 139
  • 一、基本数据类型 注释 单行注释:// 区域注释:/* */ 文档注释:/** */ 数值 对于byte类型而言...
    龙猫小爷阅读 4,257评论 0 16
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,934评论 6 13
  • 原创2016-12-28让你大吃一斤的西瓜求职 你一定记得,读书时期被迫写下过多少个关于梦想的文章。尤其是高三、考...
    大西瓜喵阅读 276评论 0 0