Java对象、对象引用关系及参数传递讨论

学习Java的过程中,很容易将对象及对象引用混为一谈,而且我们平时在交流时为了方便起见也只是对象怎样、对象怎样,但很多时候我们操作的,根本上来说是对象,但直接操作的却是对象的引用。

先贴一篇博文,非常形象的解释了对象及对象引用间的关系

  • .Java语言中通过对象的引用来操纵对象,要注意的一点是对象和对象的引用是存储在不同的地方的,因为对象往往所占的空间要求比较大,所以对象存储在堆中,而为了便于操作,对象的引用则存储在堆栈中。
  • Object obj = new Object();通过这个动作在Java中可以创建一个对象,其实这包含了4个动作:

    • new Object,以Object类为模板,在堆中创建一个对象
    • new Object(),调用Object类的构造方法,对刚生成的对象进行初始化
    • 左边的”Object obj“则是创建了一个Object类的引用变量,即obj变量是可以指向Object对象的对象引用
    • “=”操作符将对象引用obj指向刚刚创建的那个对象
  • 可以近似的将Java中对象的引用理解为C语言中的指针,即对象引用在堆栈中存储的是它所指向的对象在堆中的地址。

  • Java中的基本类型,不是通过new来创建变量,尽管基本类型变量存储在堆栈中,但它并不是引用型变量,这个变量直接存储“值”,因此更加高效。此外,由于Java是需要面向对象的,所以基本类型都会有对应的包装类。对这些包装类的值操作实际上都是通过对其对应的基本类型操作而实现的。

包装类的用途主要包含两种:
a、作为和基本数据类型对应的类类型存在,方便涉及到对象的操作。
b、包含每种基本数据类型的相关属性如最大值、最小值等,以及相关的操作方法。
c、注意Java的基本类型及其对应的包装器类型所占存储空间的大小,并不像其他大多数语言那样随机器硬件架构的变化而变化,这就使得Java程序更便于移植。

  • Java中对象的引用是存储在堆栈中的,然而对象是存储在堆中的,由于对象是通过对象的引用进行操作的,所以如果一旦一个对象没有指向它的引用变量,这个对象就成为“垃圾”,需要JVM进行垃圾回收。

  • Java只有一种参数传递方式:那就是按值传递,即Java中传递任何东西都是传值。如果传入方法的是基本类型(或其对应包类型)的东西,你就得到此基本类型的一份拷贝。如果是传递引用,就得到引用的拷贝,即指向同一块地址空间。
    Java中的基本类型及其对应包类型:
    这里写图片描述
    这些类型的变量的参数传递都是值传递,此外尽管String类型不是基本类型,但可以将其当做是char[]的包装类,这也就解释了为什么String类型的变量在方法参数传递时是值传递的原因。这也就是为什么当对字符串的操作在通过不同方法来实现的时候,推荐大家使用StringBuffer的真正原因了。

public class Main {

    public static void op1(StringBuffer s){
        s.append(" world!");
        System.out.println("s:"+s);
    }
    public static void op2(StringBuffer s){
        s = new StringBuffer("java");
        System.out.println("s:"+s);
    }
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        StringBuffer str;
        str = new StringBuffer("hello");
        System.out.println("str:"+str);
        op1(str);
        System.out.println("str:"+str);
        op2(str);
        System.out.println("str:"+str);
    }

}

要理解Java中值传递的运行机制,最好从Java内存空间分配的角度去理解,毕竟对象和对象的引用是存储在不同位置的,所以Java对象和引用的关系可以说是互相关联,却又彼此独立。彼此独立主要表现在:引用是可以改变的,它可以指向别的对象。

运行结果:
运行结果

分析:

op1操作:
这里写图片描述

op2操作:str,s是两个独立的引用变量,s是Java在内存中新建立的变量,将s指向“Java”时,原先str仍然指向“hello world”。
这里写图片描述
参数传递时,如果是对象的引用,则拷贝对象引用,使其指向相同的地址空间。

public class Main {

    public static void op1(String s){
        s += " world!";
        System.out.println("s:"+s);
    }
    public static void op2(String s){
        s = new String("java");
        System.out.println("s:"+s);
    }
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        String str;
        str = new String("hello");
        System.out.println("str:"+str);
        op1(str);
        System.out.println("str:"+str);
        op2(str);
        System.out.println("str:"+str);
    }

}

运行结果:
这里写图片描述

分析:可以看到外部方法并没有对str产生影响,因为Java语言规定String是不可变的,对String的任何操作都是产生一个新的对象。
这里写图片描述

在JAVA里,“=”不能被看成是一个赋值语句,它不是在把一个对象赋给另外一个对象,它的执行过程实质上是将右边对象的地址传给了左边的引用,使得左边的引用指向了右边的对象。JAVA表面上看起来没有指针,但它的引用其实质就是一个指针,引用里面存放的并不是对象,而是该对象的地址,使得该引用指向了对象。在JAVA里,“=”语句不应该被翻译成赋值语句,因为它所执行的确实不是一个赋值的过程,而是一个传地址的过程,被译成赋值语句会造成很多误解,译得不准确。
综上,在初始化时,“=”语句左边的是引用,右边new出来的是对象。
在后面的左右都是引用的“=”语句时,左右的引用同时指向了右边引用所指向的对象。
在参数传递时,尤其是传递的是对象的引用时,要注意“形参”和“实参”是独立的,形参在堆栈中是确实存在的,所以如果在调用方法中让形参指向了另外一个对象,实参仍然指向原来的对象。null也是一个“对象”,在Java中让形参值为null,实参的值并不等于null。

Java只有一种参数传递方式:那就是按值传递,即Java中传递任何东西都是传值。如果传入方法的是基本类型的东西,你就得到此基本类型的一份拷贝。如果是传递引用,就得到引用的拷贝,而接收这个引用的变量(形参),虽然是和实参一样指向同一块内存空间的,对指向对象操作后,其他变量访问时会得到同样的结果,但如果在此外部方法中,形参指向了另外一个对象,那么形参和实参就是独立的了,接下来的操作不会对另外一个产生任何的影响!例如,如果要是将想将某个变量置为null的话,千万不要通过类的方法setNull实现,因为那样只是让形参置值为null,不会对实参产生任何的影响,置null直接让其赋值为null即可。

  • 4
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1. 简单型是按值传递的   Java 方法的参数是简单型的时候,是按值传递的 (pass by value)。这一点我们可以通过一个简单的例子来说明: /* 例 1 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void test(boolean test) { test = ! test; System.out.println("In test(boolean) : test = " + test); } public static void main(String[] args) { boolean test = true; System.out.println("Before test(boolean) : test = " + test); test(test); System.out.println("After test(boolean) : test = " + test); } } 运行结果: Before test(boolean) : test = true In test(boolean) : test = false After test(boolean) : test = true   不难看出,虽然在 test(boolean) 方法中改变了传进来的参数的值,但对这个参数源变量本身并没有影响,即对 main(String[]) 方法里的 test 变量没有影响。那说明,参数型是简单型的时候,是按值传递的。以参数形式传递简单型的变量时,实际上是将参数的值作了一个拷贝传进方法函数的,那么在方法函数里再怎么改变其值,其结果都是只改变了拷贝的值,而不是源值。   2. 什么是引用   Java传值还是传引用,问题主要出在对象的传递上,因为 Java 中简单型没有引用。既然争论中提到了引用这个东西,为了搞清楚这个问题,我们必须要知道引用是什么。   简单的说,引用其实就像是一个对象的名字或者别名 (alias),一个对象在内存中会请求一块空间来保存数据,根据对象的大小,它可能需要占用的空间大小也不等。访问对象的时候,我们不会直接是访问对象在内存中的数据,而是通过引用去访问。引用也是一种数据型,我们可以把它想象为似 C 语言中指针的东西,它指示了对象在内存中的地址——只不过我们不能够观察到这个地址究竟是什么。   如果我们定义了不止一个引用指向同一个对象,那么这些引用是不相同的,因为引用也是一种数据型,需要一定的内存空间来保存。但是它们的值是相同的,都指示同一个对象在内存的中位置。比如 String a = "Hello"; String b = a;   这里,a 和 b 是不同的两个引用,我们使用了两个定义语句来定义它们。但它们的值是一样的,都指向同一个对象 "Hello"。也许你还觉得不够直观,因为 String 对象的值本身是不可更改的 (像 b = "World"; b = a; 这种情况不是改变了 "World" 这一对象的值,而是改变了它的引用 b 的值使之指向了另一个 String 对象 a)。那么我们用 StringBuffer 来举一个例子: /* 例 2 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void main(String[] args) { StringBuffer a = new StringBuffer("Hello"); StringBuffer b = a; b.append(", World"); System.out.println("a is " + a); } } 运行结果: a is Hello, World   这个例子中 a 和 b 都是引用,当改变了 b 指示的对象的值的时候,从输出结果来看,a 所指示的对象的值也改变了。所以,a 和 b 都指向同一个对象即包含 "Hello" 的一个 StringBuffer 对象。   这里我描述了两个要点: 引用是一种数据型,保存了对象在内存中的地址,这种型即不是我们平时所说的简单数据型也不是实例(对象); 不同的引用可能指向同一个对象,换句话说,一个对象可以有多个引用,即该型的变量。   3. 对象是如何传递的呢   关于对象的传递,有两种说法,即“它是按值传递的”和“它是按引用传递的”。这两种说法各有各的道理,但是它们都没有从本质上去分析,即致于产生了争论。   既然现在我们已经知道了引用是什么东西,那么现在不妨来分析一下对象作是参数是如何传递的。还是先以一个程序为例: /* 例 3 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void test(StringBuffer str) { str.append(", World!"); } public static void main(String[] args) { StringBuffer string = new StringBuffer("Hello"); test(string); System.out.println(string); } } 运行结果:   Hello, World!   test(string) 调用了 test(StringBuffer) 方法,并将 string 作为参数传递了进去。这里 string 是一个引用,这一点是勿庸置疑的。前面提到,引用是一种数据型,而且不是对象,所以它不可能按引用传递,所以它是按值传递的,它么它的值究竟是什么呢?是对象的地址。   由此可见,对象作为参数的时候是按值传递的,对吗?错!为什么错,让我们看另一个例子: /* 例 4 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void test(String str) { str = "World"; } public static void main(String[] args) { String string = "Hello"; test(string); System.out.println(string); } }   运行结果: Hello   为什么会这样呢?因为参数 str 是一个引用,而且它与 string 是不同的引用,虽然它们都是同一个对象引用。str = "World" 则改变了 str 的值,使之指向了另一个对象,然而 str 指向的对象改变了,但它并没有对 "Hello" 造成任何影响,而且由于 string 和 str 是不同的引用,str 的改变也没有对 string 造成任何影响,结果就如例中所示。   其结果是推翻了参数按值传递的说法。那么,对象作为参数的时候是按引用传递的了?也错!因为上一个例子的确能够说明它是按值传递的。   结果,就像光到底是波还是粒子的问题一样,Java 方法的参数是按什么传递的问题,其答案就只能是:即是按值传递也是按引用传递,只是参照物不同,结果也就不同。   4. 正确看待传值还是传引用的问题   要正确的看待这个问题必须要搞清楚为什么会有这样一个问题。   实际上,问题来源于 C,而不是 Java。   C 语言中有一种数据型叫做指针,于是将一个数据作为参数传递给某个函数的时候,就有两种方式:传值,或是传指针,它们的区别,可以用一个简单的例子说明: /* 例 5 */ /** * @(#) test.c * @author fancy */ void SwapValue(int a, int b) { int t = a; a = b; b = t; } void SwapPointer(int * a, int * b) { int t = * a; * a = * b; * b = t; } void main() { int a = 0, b = 1; printf("1 : a = %d, b = %d/n", a, b); SwapValue(a, b); printf("2 : a = %d, b = %d/n", a, b); SwapPointer(&a, &b); printf("3 : a = %d, b = %d/n", a, b); } 运行结果: 1 : a = 0, b = 1 2 : a = 0, b = 1 3 : a = 1, b = 0   大家可以明显的看到,按指针传递参数可以方便的修改通过参数传递进来的值,而按值传递就不行。   当 Java 成长起来的时候,许多的 C 程序员开始转向学习 Java,他们发现,使用似 SwapValue 的方法仍然不能改变通过参数传递进来的简单数据型的值,但是如果是一个对象,则可能将其成员随意更改。于是他们觉得这很像是 C 语言传值/传指针的问题。但是 Java 中没有指针,那么这个问题就演变成了传值/传引用的问题。可惜将这个问题放在 Java 中进行讨论并不恰当。   讨论这样一个问题的最终目的只是为了搞清楚何种情况才能在方法函数中方便的更改参数的值并使之长期有效。   Java 中,改变参数的值有两种情况,第一种,使用赋值号“=”直接进行赋值使其改变,如例 1 和例 4;第二种,对于某些对象引用,通过一定途径对其成员数据进行改变,如例 3。对于第一种情况,其改变不会影响到方法该方法以外的数据,或者直接说源数据。而第二种方法,则相反,会影响到源数据——因为引用指示的对象没有变,对其成员数据进行改变则实质上是改变的该对象。   5. 如何实现似 swap 的方法   传值还是传引用的问题,到此已经算是解决了,但是我们仍然不能解决这样一个问题:如果我有两个 int 型的变量 a 和 b,我想写一个方法来交换它们的值,应该怎么办?   结论很让人失望——没有办法!因此,我们只能具体情况具体讨论,以经常使用交换方法的排序为例: /** 例 6 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void swap(int[] data, int a, int b) { int t = data[a]; data[a] = data[b]; data[b] = t; } public static void main(String[] args) { int[] data = new int[10]; for (int i = 0; i < 10; i++) { data[i] = (int) (Math.random() * 100); System.out.print(" " + data[i]); } System.out.println(); for (int i = 0; i < 9; i++) { for (int j = i; j < 10; j++) { if (data[i] > data[j]) { swap(data, i, j); } } } for (int i = 0; i < 10; i++) { System.out.print(" " + data[i]); } System.out.println(); } } 运行结果(情况之一): 78 69 94 38 95 31 50 97 84 1 1 31 38 50 69 78 84 94 95 97   swap(int[] data, int a, int b) 方法在内部实际上是改变了 data 所指示的对象的成员数据,即上述讨论的第二种改变参数值的方法。希望大家能够举一反三,使用似的方法来解决相关问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值