程序员

Java中的参数传递

2017-12-22  本文已影响93人  AmorFatiYJ

为了便于理解,会将参数传递分为按值传递和按引用传递。按值传递是传递的值的拷贝,按引用传递传递的是引用的地址值,所以有“在java里参数传递都是按值传递”、“引用也是按值传递的”这些说法。

基本数据类型作为参数传递

基本数据类型作为参数传递时都是传递值的拷贝。 指的是方法调用中,传递的是值的拷贝,无论怎么改变这个拷贝,原值是不会改变的。

也即是说实参把它的值传递给形参,对形参的改变不会影响实参的值。

public class Test {
    public static void main(String[] args) {
        Test test1 = new Test();
        int i = 5;
        System.out.println("调用前的i=" + i);
        test1.testPassParameter(i);
        //传递后,testPassParameter方法中对形参i的改变不会影响这里的i
        System.out.println("调用后的i=" + i);
    }
    public void testPassParameter(int i) {
        i = 10;//这里只是对形参的改变
        System.out.println("tpp方法中的i=" + i);
    }
}

输出结果

调用前的i=5
tpp方法中的i=10
调用后的i=5

对象作为参数传递

对象作为参数传递时,在方法内改变对象的值,有时原对象跟着改变,而有时又没有改变。就会让人对“传值”和“传引用”产生疑惑。

先看看两种情况的例子。

public class Test {
    public static void main(String[] args) {
        StringBuffer s1 = new StringBuffer("hello");
        StringBuffer s2 = new StringBuffer("hello");
        changeStringBuffer(s1, s2);
        System.out.println("s1=" + s1);
        System.out.println("s2=" + s2);
    }
    public static void changeStringBuffer(StringBuffer ss1, StringBuffer ss2) {
        ss1.append("world");
        ss2 = ss1;
    }
}

输出结果:

s1=helloworld  
s2=hello

分析如下:

//(1)s1,s2指向字符串的地址不同,假设为地址1,地址2
StringBuffer s1 = new StringBuffer("hello");
StringBuffer s2 = new StringBuffer("hello");
image
//(2)调用changeStringBuffer方法
changeStringBuffer(s1, s2);
//调用后会将s1,s2的地址(地址1,地址2)传给ss1,ss2,即现在ss1也指向地址1,ss2也指向地址2
image
//(3)ss1所指向字符串的值变为helloworld,调用者s1的值相应变化。
ss1.append("world");
//ss2将指向ss1指向的地址(地址1),但此时s2依旧指向地址2,s2的值在调用前后不变。
ss2 = ss1;
image

所以,s1的输出为helloworld,s2的输出结果为hello。

(注意,这里为了区分,s1,s2和ss1,ss2用了不同的名称,但有时候形参与实参的名字相同,其实两者变量是完全不同的,一个是main方法中的变量,一个是changeStringBuffer()中的变量。)

可以看出,在java中对象作为参数传递时,传递的是引用的地址,是把对象在内存中的地址拷贝了一份传给了参数

拓展:

基本数据类型的包装类型在传递参数时其实也是“按引用传递的”,只是因为包装类型变量都是不可变量,容易误解。

String是final类型,是个特殊的类,对它的一些操作符是重载的。比如:

String str = "hello";//等价于String str=new String("hello");

 String str = "hello";
        str = str + "world";//等价于str = new String(new StringBuffer(str).append("world"));

从以上分析,现在可以更易理解下面的代码:

public class Test {
    public static void fun(String str, char ch[]) {
        str = "world";
        ch[0] = 'd';
    }
    public static void main(String[] args) {
        String str = new String("hello");
        char[] ch = {'a', 'b', 'c'};
        fun(str, ch);
        System.out.println(str);
        System.out.println(ch);
    }
}

输出结果

hello  
dbc

方法调用时,名称相同的实参和形参并不一样,一个是main()中的str,指向存放"hello"的内存地址。一个是fun()中的str,str="world",相当于new String("world")。String是final类型,将在堆中重新分配一个内存空间存放"world"。ch[0]='d',对象的内容发生改变。

image

所以main()中str变量存放的对象内容依然是"hello"。ch从"abc"变为"dbc"。

参考:

上一篇 下一篇

猜你喜欢

热点阅读