您的位置:首页 > 编程语言 > Java开发

java中值传递与引用传递

2018-01-07 23:08 288 查看
我们首先来区分一下什么是对象与对象引用:

1. 何谓对象?

  在Java中有一句比较流行的话,叫做“万物皆对象”,这是Java语言设计之初的理念之一。要理解什么是对象,需要跟类一起结合起来理解。下面这段话引自《Java编程思想》中的一段原话:

  “按照通俗的说法,每个对象都是某个类(class)的一个实例(instance),这里,‘类’就是‘类型’的同义词。”

  从这一句话就可以理解到对象的本质,简而言之,它就是类的实例,比如所有的人统称为“人类”,这里的“人类”就是一个类(物种的一种类型),而具体到每个人,比如张三这个人,它就是对象,就是“人类”的实例。

2. 何谓对象引用?
我们先看一段话:

  “每种编程语言都有自己的数据处理方式。有些时候,程序员必须注意将要处理的数据是什么类型。你是直接操纵元素,还是用某种基于特殊语法的间接表示(例如C/C++里的指针)来操作对象。所有这些在 Java 里都得到了简化,一切都被视为对象。因此,我们可采用一种统一的语法。尽管将一切都“看作”对象,但操纵的标识符实际是指向一个对象的“引用”(reference)。”  

  这段话来自于《Java编程思想》,很显然,从这段话可以看出对象和对象引用不是一回事,是两个完全不同的概念。举个例子,我们通常会用下面这一行代码来创建一个对象:

Person person = new Person("张三");


  有人会说,这里的person是一个对象,是Person类的一个实例。

  也有人会说,这里的person并不是真正的对象,而是指向所创建的对象的引用。

  到底哪种说法是对的?我们先不急着纠结哪种说法是对的,再看两行代码:

Person person;
person = new Person("张三");


  这两行代码实现的功能和上面的一行代码是完全一样的。大家都知道,在Java中new是用来在堆上创建对象用的,如果person是一个对象的话,那么第二行为何还要通过new来创建对象呢?由此可见,person并不是所创建的对象,是什么?上面的一段话说的很清楚,“操纵的标识符实际是指向一个对象的引用”,也就是说person是一个引用,是指向一个可以指向Person类的对象的引用。真正创建对象的语句是右边的new Person("张三");

  再看一个例子:

Person person;
person = new Person("张三");
person = new Person("李四");


  这里让person先指向了“张三”这个对象,然后又指向了“李四”这个对象。也就是说,Person person,这句话只是声明了一个Person类的引用,它可以指向任何Person类的实例。这个道理就和下面这段代码一样:

int a;
a=2;
a=3;


  这里先声明了一个int类型的变量a,先对a赋值为2,后面又赋值为3.也就是说int类型的变量a,可以让它的值为2,也可以为3,只要是合法的int类型的数值即可。

  也就是说,一个引用可以指向多个对象,而一个对象可不可以被多个引用所指呢?答案当然是可以的。

  比如:

Person person1 = new Person("张三");
Person person2 = person1;


  person1和person2都指向了“张三”这个对象。

2. 值传递与引用传递

理解了上面概念我们可以跟好的理解,什么是值传递,什么是引用传递。

Java中只有按值传递。不管是基本类型还是引用类型,形参都是实参的一个拷贝。对形参的修改不会影响到实参。形参只是实参的一个拷贝,是两个不同的东西。但是因为形参和实参的值相同,都指向同一个对象,所以对[b]形参所指向的对象的修改会影响到实参所指向的对象
[/b]

形参:用来接收调用该方法时传递的参数。只有在被调用的时候才分配内存空间,一旦调用结束,就释放内存空间。因此仅仅在方法内有效。

实参:传递给被调用方法的值,预先创建并赋予确定值。

值传递:方法调用时,实际参数把它的值的副本传递给对应的形式参数。特点:此时内存中存在两个相等的基本类型,即实际参数和形式参数后面方法中的操作都是对形参这个值的修改,不影响实际参数的值

引用传递:方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,函数接收的是原始值的内存地址;特点:在方法执行中,形参和实参内容相同,指向同一块内存地址,方法执行中[b]对引用的操作将会影响到实际对象[/b]。

传引用,方法体内改变形参引用,不会改变实参的引用,但有可能改变实参对象的属性值。

举两个例子:
(1)方法体内改变形参引用,但不会改变实参引用 ,实参值不变。

public class TestFun2 {
        public static void testStr(String str) {
            str = "hello";// 型参指向字符串 “hello”
        }

        public static void main(String[] args) {
            String s = "1";
            TestFun2.testStr(s);
            System.out.println("s=" + s); // 实参s引用没变,值也不变
        }
    }

(2)方法体内,通过引用改变了实际参数对象的内容,注意是“内容”,引用还是不变的。

public class TestFun4 {
public static void testStringBuffer(StringBuffer sb) {
sb.append("java");// 改变了实参的内容
}

public static void main(String[] args) {
StringBuffer sb = new StringBuffer("my ");
new TestFun4().testStringBuffer(sb);
System.out.println("sb=" + sb.toString());// 内容变化了
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: