zoukankan      html  css  js  c++  java
  • java初步—参数的值传递

         校招季,本人匆匆忙忙地参加各种宣讲会,几次笔试下来都遇到同一个题目,而且全都错在同一想法上,方知自己的基础实在不太牢固,因此特别写在博客上提醒自己要脚踏实地地学习!不多说了,题目如下:

     1 public class Test {
     2     public static void main(String[] args) {
     3         StringBuffer s1 = new StringBuffer("a");
     4         StringBuffer s2 = new StringBuffer("b");
     5         change(s1, s2);
     6         System.out.println(s1);
     7         System.out.println(s2);
     8     }
     9 
    10     private static void change(StringBuffer s1, StringBuffer s2) {
    11         s1.append(s2);
    12         s2=s1;
    13     }
    14 }

          问题:请问最后打印出的结果是什么?

          如果阅读我这篇随笔的是刚接触java(c#也行,java和C#在这方面基本一致)的同学,不妨不看后面的答案,自己做一下,也许会给你带来一些收获!


          我在笔试的时候有四个选项:

          A.a,b    B.ab,b    C.a,ab    D.ab,ab

          很可能有不少人都会和我一样直接选择了选项D,不过很抱歉,这是一个绝对错误的答案,正确的答案是  B   选项。

          是不是也会有人和我刚开始一样觉得很奇怪呢?因为根据我们学到的知识,StringBuffer类型的变量是引用类型,也就意味着这个变量在change方法中的改变是会被保留下来的,那么s1,s2在change方法中都被改变了,为什么最后打印的时候只有s1是真正被改变了呢?

          这个需要先知道一些基本的知识:

          我们都知道在java中有基本数据类型和引用数据类型,他们在内存中的存储是不同的,基本数据类型诸如int,double之类的变量是直接把值存放在栈中的,而引用数据类型的变量是分为两部分进行存储的:第一部分和基本数据类型一致是存放在栈中的,第二部分是堆中。其中堆中存放的是这个变量真正的值,而栈中则是存放真正值的地址,就上面的题目来说,示意图如下:

    图一

          好了,如果知道变量的存储方式的话就可以往下了(如果不知道建议看看这方面专门的文章,我这篇说的不细)。

          change方法中的参数名也是这题的障碍所在,因为名字和变量相同,会让人很容易进入陷阱,误认change方法中的参数s1,s1就是main方法中的变量s1,s2,我们不妨把参数名改一下,改后的代码如下:

     1 public class Test {
     2     public static void main(String[] args) {
     3         StringBuffer s1 = new StringBuffer("a");
     4         StringBuffer s2 = new StringBuffer("b");
     5         change(s1, s2);
     6         System.out.println(s1);
     7         System.out.println(s2);
     8     }
     9 
    10     private static void change(StringBuffer a, StringBuffer b) {
    11         a.append(b);
    12         b=a;
    13     }
    14 }

          注意第10行红色加粗的参数名,这样我们就可以绕开一点障眼法。

          其实在这题中还是有个小问题的,那就是java中到底有没有引用传递的问题,如果是基本数据类型的话,如

    1 int x=10;
    2 
    3 print(x);
    4 
    5 public void print(int y){
    6    System.out.println(y++);
    7 }

          此时x作为print方法的参数,传递的是x真正的值,我们很容易知道在print方法中y的值是x值的复制,也就是说变量x只是把自己的值拷贝了一份给另外一个局部变量y,实质上二者没有任何关系,无论在print方法中y如何变化都不会影响x的值,这就是值传递了。而如果int x,y的类型改为Integer,此时变量都变为引用类型,x作为参数传递的就是x值的地址,那么y值在print方法中的改变就会反映到x的改变,这就是引用传递。

          仔细想想,值传递和引用传递其实传递的都是作为参数的直接值(引用类型变量的直接值是真正值的地址),因此实质上说java只有值传递更为恰当。

          解决了这个比较主观的问题之后,我们再来研究一下,引用类型变量s1,s2把地址传递给change方法之后,局部变量a,b就成为了s1,s2的别名(拥有相同的地址,指向了同一实例对象,只是名称不同而已,就和爸妈与老师叫你的称呼不一样),s1追加了s2的内容,所以变量s1的变化会被保留下来,最终打印的结果自然是“ab”,没有疑问。

          重点来了,执行“a=b;”进行了变量的什么操作呢?

    图二

          这样就可以清晰地看到,"b=a"这个操作实质只是把局部变量a的直接值(也就是真正值的地址)赋给了变量b,对于变量b的真正值没有任何的操作,所以对应变量s2的真正值没有任何变化,自然最后打印出来的结果是 “b".

     

  • 相关阅读:
    Python
    Linux, Nginx
    Python
    C#图像处理(各种旋转、改变大小、柔化、锐化、雾化、底片、浮雕、黑白、滤镜效果)
    堆——神奇的优先队列(下)
    堆——神奇的优先队列(上)
    二叉树
    开启“树”之旅
    巧妙的邻接表(数组实现)
    Dijkstra最短路算法
  • 原文地址:https://www.cnblogs.com/liaochong/p/thinkinjava.html
Copyright © 2011-2022 走看看