在Java中,方法的参数的传递分为值传递(基本数据)和引用传递(引用数据:对象、字符串),这是最容易接受的。如果你能知道有这两种情况存在,那么,在遇到调用方法时,你可以避免很多问题的产生。但是,仔细查阅资料发现,Java中只有值传递。那么你会问,那为什么还叫引用传递呢?这让我想弄明白到底,方法执行过程中,参数是怎么使用的。
Java只有值传递
这里我准备了一个事例用来分析,如下:
@Testpublic voidTest2() {
A a= new A("apple");int b = 2;
System.out.println();
System.out.println("方法执行前:");
System.out.println(a.toString());
System.out.println(b);
testParam(a, b);
System.out.println();
System.out.println("方法执行后:");
System.out.println(a.toString());
System.out.println(b);
}private void testParam(A a1, intb1) {
a1.setName("banana");
b1= 3;
System.out.println();
System.out.println("方法中:");
System.out.println(a1.toString());
System.out.println(b1);
}classA {privateString name;publicA(String name) {this.name =name;
}publicString getName() {returnname;
}public voidsetName(String name) {this.name =name;
}
@OverridepublicString toString() {return "A{" +
"name='" + name + '\'' +
'}';
}
}
输入结果:
1:首先,所有的方法在调用时,参数都会进行拷贝。
想理解这句话,我觉得很有必要先弄懂 栈内存 和 堆内存。首先,java中的基本数据类型 和 指针(指针是指向一块内存地址的内存数据,也就是说指针本身是一个占用4字节内存的int(32 位系统内))都是存放在 栈内存 中的。所以,当方法被调用时,基本数据类型 和 指针 都进行了拷贝。
也就是说,引用传递实际上也是对栈内存中的指针进行拷贝。
所谓传引用的说法是为了更好的区别基本数据类型,理解其调用方式。基于上面对指针的理解,我们不难看出,指针其实也是一个int值,所谓传引用,我们是复制了指针的int值进行传递。为了便于理解,我们可以姑且把指针看作一种数据类型,透明化指针的int特性,从而提出传引用的概念。
在Java中,很多人说没有指针,事实上,在Java更深层次里,到处都是大师封装好的精美绝伦的指针。为了更容易的讲解Java中关于类和类型的调用,Java中出现了值与引用的说法。浅显的来说,我们可以认为Java中的引用与C中的指针等效。
引用传递:当使用引用变量作为形参时,它将变为实参列表中相应变量的别名,对形参进行的任何更改都将真正更改正在调用它的函数中的变量。当以这种方式将数据传递给形参时,该实参被称为按引用传递。
下面我们通过一个反例来说明:「Java不是引用传递」
classB {privateString name;privateInteger age;publicB(String name, Integer age) {this.name =name;this.age =age;
}
@OverridepublicString toString() {return "B{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}public static voidexchange(B b3, B b4) {
B b5=b3;
b3=b4;
b4=b5;
}public static voidmain(String[] args) {
B b1= new B("张三", 24);
B b2= new B("李四", 13);
System.out.println("调用前");
System.out.println(b1.toString());
System.out.println(b2.toString());
exchange(b1, b2);
System.out.println("调用后");
System.out.println(b1.toString());
System.out.println(b2.toString());
}
}
打印结果是:
我们发现b1和b2的值没有变化,我们只是将拷贝的b3和b4的值进行了交换,b1和b2引用仍然指向之前的对象。
加餐
提问:如果传一个String字符串作为参数,并在方法中对字符串参数进行修改。站在 String字符串 是引用数据类型的前提下,那么你认为,在方法调用前和调用后,值会是一致的吗?
@Testpublic voidTest2() {
String str= "1111";
System.out.println("方法执行前:");
System.out.println(str);
testString(str);
System.out.println();
System.out.println("方法执行后:");
System.out.println(str);
}private voidtestString(String str1) {
str1+= "2222";
System.out.println();
System.out.println("方法中:");
System.out.println(str1);
}
如果你回答不一致,那么只能说,上面的你确实听懂了。不过,字符串这块的知识你需要再更深入的学习了。╰( ̄▽ ̄)╮╰( ̄▽ ̄)╮╰( ̄▽ ̄)╮
输入结果:
咦,怎么字符串str没有因为参数字符串str1的改变而改变呢?难道,参数str1没有修改str的堆内存内容吗,往这方面思考是对的,它确实没有修改。
这里需要讲一个我们听过最多的,但是在用的时候又想不起来的知识点:字符串不能修改。
通过反编译字节码文件,可以看到,字符串在进行 + 运算符时,新建了一个 StringBuild对象,通过 该对象的 append() 方法进行字符串的拼接,然后通过 toString() 方法返回一个新的堆内存地址。所以str1 指向的是新的堆内存地址。
终于,对于Java 参数的传递问题,应该都明白了。( ^_^ )/~~拜拜
欢迎大家评论与交流,加油!!
【参考】