java 方法中交换Integer类型的局部变量a,b

  自己的第一篇博客,各位看官多多指教。这里讲的是一道面试题,题目如下:

    //要求写一个swap方法交换ab值输出打印a=2 b=1
    public static void main(String[] args) {
        Integer a=1,b=2;
        swap(a,b);
        System.out.println("a="+a+"   b="+b);
    }

  这道题目所涉及到的知识点包括:

  • Integer的自动装箱
  • java的值传递和引用传递
  • 对Integer类的源码分析
  • java的反射机制
1. Integer的自动装箱

  java的自动装箱就是自动将原始数据类型(byte,short,char,int,long,float,double,boolean)转换成对应的基本数据类型包装类对象(Byte,Short,Character,Integer,Long,Float,Double,Boolean),比如将int的变量转换成Integer对象
这里的Integer a=1 经过自动装箱以后变为
Integer a=Integer.valueOf(1);
那么我们接着查看Integer类中的valueOf()方法如下:

    public static Integer valueOf(int i) {
        if (i >= IntegerCache.low && i <= IntegerCache.high)
            return IntegerCache.cache[i + (-IntegerCache.low)];
        return new Integer(i);
    }

  接着查看IntegerCache这个内部类可以发现java虚拟机在我们首次使用Integer的时候会初始化数值在-128-127之间的整数,而且这个区间范围可以在虚拟机启动的时候自己设定。换句话说,数值在-128-127之间的相同整数都会取自相同的内存具体值。下面例子中a和b都是指向相同的内存地址,而x和y指向两块不同的内存地址。

    Integer x=-129;
    Integer y=-129;
    System.out.println(x==y);    //输出false
    Integer a=-128;
    Integer b=-128;
    System.out.println(a==b);    //输出true
    Integer xx=new Integer(1);
    Integer yy=new Integer(1);
    System.out.println(xx==yy);    //输出false

  说到这里必须说一下valueOf(int i)方法最终返回的是new Integer(i);而Integer类的构造函数如下,这里的value是解题的关键,value在Integer类中是一个final的私有成员变量。

    public Integer(int value) {
        this.value = value;
    }

说出正确答案之前还得看看Integer的源码中的toString方法
public String toString() { return toString(value); }
可见其返回的真实值就是value,并且value就是在构造函数中初始化的。在swap方法中我们拿到了Integer的引用,那么我们就可以通过反射来改变对应实例的局部变量值。具体代码如下:

    private static void swap1(Integer aa, Integer bb) {
        try {
            Field aaValue = aa.getClass().getDeclaredField("value");
            aaValue.setAccessible(true);
            aaValue.set(aa,2);
            Field bbValue = bb.getClass().getDeclaredField("value");
            bbValue.setAccessible(true);
            bbValue.set(bb,1);
            System.out.println(Integer.valueOf(1));//输出2
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

但是运行之后我们发现其结果是:

a=2   b=2

  看到这里是不是感觉很不可思议,但是我们距离真相已经很近了。我们疑惑的是为什么b的值没有发生改变,更疑惑的是为什么Integer.valueOf(1)输出的竟然是2。这是因为基本数据类型在-128-127之间的数都会自动装箱并且从缓存中去取,我们反射改变的只是缓存中的1对应的实例的实例变量value值,我们在bbValue.set(bb,1);的时候1会自动装箱去缓存中去找,而这时缓存中的Integer.valueOf(1)已经是2,所以返回的还是2。解决的方法就是不让其去缓存中找,参考代码如下:

    //正解
    private static void swap1(Integer aa, Integer bb) {
        try {
            Field aaValue = aa.getClass().getDeclaredField("value");
            aaValue.setAccessible(true);
            aaValue.set(aa,2);
            Field bbValue = bb.getClass().getDeclaredField("value");
            bbValue.setAccessible(true);
            //这里的1不会自动装箱,当然就不会从缓存中去取值
            bbValue.set(bb,new Integer(1));
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

  这里还是要多说一句,如果a,b变量的值是-128-127之外的整数的话虽然会自动装箱,但是不会从缓存中去取值,那么bbValue.set(bb,-128-127之外的整数);就可以了,是不用专门new Integer()的,参考代码如下:

    public static void main(String[] args) {
        Integer a=999;
        Integer b=888;
        swap1(a,b);
        System.out.println("a="+a+"   b="+b);
    }
    private static void swap1(Integer aa, Integer bb) {
        try {
            Field aaValue = aa.getClass().getDeclaredField("value");
            aaValue.setAccessible(true);
            aaValue.set(aa,888);
            Field bbValue = bb.getClass().getDeclaredField("value");
            bbValue.setAccessible(true);
            bbValue.set(bb,999);//这里没有new Integer(999)
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

输出结果为a=888 b=999

2. 错误答案示例

  对于这道题目的解法有些同学可能会给出下面的答案,明显是错误的,因为它们具体交换的还是aa和bb所指向的内存地址值,而这一块内存地址值所指向的内存空间的内容并没有改变,所以a和b变量是没有变化的(参考图1)。

    private static void swap(Integer aa,Integer bb){
        Integer temp;
        temp=aa;
        bb=temp;
        aa=bb;
    }

  按照我自己的理解,java的值传递和引用传递实际上都是值传递,因为引用传递最终传递的还是引用对象的内存具体值(下图红色线条代表交换后的状态,可见a和b并没有发生变化)


图1.引用传递(红线代表交换后)
3. 题目变种
    //对于这道题是不能用反射的,因为是基本数据类型的值传递
    public static void main(String[] args) {
        int a=1;
        int b=2;
        swap(a,b);
        System.out.println("a="+a+"   b="+b);
    }

投机取巧的做法如下:

    public static void main(String[] args) {
        int a=1;
        int b=2;
        swap(a,b);
        System.out.println("a="+a+"   b="+b);
    }
    private static void swap(Integer aa, Integer bb) {
        System.out.println("a="+2+"   b="+1);
        System.exit(0);
    }

  题目虽小,却包含很多知识点,希望大家能够有所收获,欢迎留言讨论!

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,652评论 18 139
  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,701评论 0 11
  • 自动装箱和拆箱从Java1.5开始引入,目的是将原始类型值转自动地转换成对应的对象。自动装箱与拆箱的机制可以让我们...
    GB_speak阅读 619评论 0 4
  • 下面的内容是对网上原有的Java面试题集及答案进行了全面修订之后给出的负责任的题目和答案,原来的题目中有很多重复题...
    独念白阅读 1,341评论 0 3
  • 十六岁的是什么样的年龄呢,最近不小心点了《最好的我们》看来一点点就不能控制自己的情绪了。原来十六岁的那年,我...
    Mrs简阅读 112评论 0 0