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