Java 中的值传递和参数传递

Java中没有指针,所以也没有引用传递了,仅仅有值传递不过可以通过对象的方式来实现引用传递 类似java没有多继承 但可以用多次implements 接口实现多继承的功能

值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 数的值。 引用传递:也称为传地址。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,方法执行中形式参数值的改变将会影响实际参数的值。

Java参数按值传递

面试题:当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?     答:是值传递。Java 编程语言只有值传递参数。当一个对象实例作为一个参数被传递到方法中时,参数的值就是该对象的引用一个副本。指向同一个对象,对象的内容可以在被调用的方法中改变,但对象的引用(不是引用的副本)是永远不会改变的。

-------------------------------------------------------------

在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。    Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。     按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数的值,调用代码中的原始值也随之改变。如果函数修改了该参数的地址,调用代码中的原始值不会改变.     当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。     Java 应用程序按值传递参数(引用类型或基本类型),其实都是传递他们的一份拷贝.而不是数据本身.(不是像 C++ 中那样对原始值进行操作。)
例1:

Java代码

  1. //在函数中传递基本数据类型,
  2. public class Test {
  3. public static void change(int i, int j) {
  4. int temp = i;
  5. i = j;
  6. j = temp;
  7. }
  8. public static void main(String[] args) {
  9. int a = 3;
  10. int b = 4;
  11. change(a, b);
  12. System.out.println("a=" + a);
  13. System.out.println("b=" + b);
  14. }
  15. }
  16. 结果为:
  17. a=3
  18. b=4
  19. 原因就是 参数中传递的是 基本类型 a 和 b 的拷贝,在函数中交换的也是那份拷贝的值 而不是数据本身;

例2:

Java代码

  1. //传的是引用数据类型
  2. public class Test {
  3. public static void change(int[] counts) {
  4. counts[0] = 6;
  5. System.out.println(counts[0]);
  6. }
  7. public static void main(String[] args) {
  8. int[] count = { 1, 2, 3, 4, 5 };
  9. change(count);
  10. }
  11. }
  12. 在方法中 传递引用数据类型int数组,实际上传递的是其引用count的拷贝,他们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。

例3:

Java代码

  1. //对象的引用(不是引用的副本)是永远不会改变的
  2. class A {
  3. int i = 0;
  4. }
  5. public class Test {
  6. public static void add(A a) {
  7. a = new A();
  8. a.i++;
  9. }
  10. public static void main(String args[]) {
  11. A a = new A();
  12. add(a);
  13. System.out.println(a.i);
  14. }
  15. }
  16. 输出结果是0
  17. 在该程序中,对象的引用指向的是A ,而在change方法中,传递的引用的一份副本则指向了一个新的OBJECT,并对其进行操作。
  18. 而原来的A对象并没有发生任何变化。 引用指向的是还是原来的A对象。

例4: String 不改变,数组改变

Java代码

  1. public class Example {
  2. String str = new String("good");
  3. char[] ch = { ‘a‘, ‘b‘, ‘c‘ };
  4. public static void main(String args[]) {
  5. Example ex = new Example();
  6. ex.change(ex.str, ex.ch);
  7. System.out.print(ex.str + " and ");
  8. System.out.println(ex.ch);
  9. }
  10. public void change(String str, char ch[]) {
  11. str = "test ok";
  12. ch[0] = ‘g‘;
  13. }
  14. }
  15. 程序3输出的是 good and gbc.
  16. String 比较特别,看过String 代码的都知道, String 是 final的。所以值是不变的。 函数中String对象引用的副本指向了另外一个新String对象,而数组对象引用的副本没有改变,而是改变对象中数据的内容.
  17. 对于对象类型,也就是Object的子类,如果你在方法中修改了它的成员的值,那个修改是生效的,方法调用结束后,它的成员是新的值,但是如果你把它指向一个其它的对象,方法调用结束后,原来对它的引用并没用指向新的对象。

Java参数,不管是原始类型还是引用类型,传递的都是副本(有另外一种说法是传值,但是说传副本更好理解吧,传值通常是相对传址而言)。

如果参数类型是原始类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个跟之前所谈的传值是一样的。如果在函数中改变了副本的值不会改变原始的值.

如果参数类型是引用类型,那么传过来的就是这个引用参数的副本,这个副本存放的是参数的地址。如果在函数中没有改变这个副本的地址,而是改变了地址中的值,那么在函数内的改变会影响到传入的参数。如果在函数中改变了副本的地址,如new一个,那么副本就指向了一个新的地址,此时传入的参数还是指向原来的地址,所以不会改变参数的值。

( 对象包括对象引用即地址和对象的内容)

a.传递值的数据类型:八种基本数据类型和String(这样理解可以,但是事实上String也是传递的地址,只是string对象和其他对象是不同的,string对象是不能被改变的,内容改变就会产生新对象。那么StringBuffer就可以了,但只是改变其内容。不能改变外部变量所指向的内存地址)。

b.传递地址值的数据类型:除String以外的所有复合数据类型,包括数组、类和接口

下面举例说明:

在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。

Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。

按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数,调用代码中的原始值也随之改变。

当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。

Java 应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。

  1. class Test
  2. {
  3. public static void main(String args[])
  4. {
  5. int val;
  6. StringBuffer sb1, sb2;
  7. val = 10;
  8. sb1 = new StringBuffer("apples");
  9. sb2 = new StringBuffer("pears");
  10. System.out.println("val is " + val);
  11. System.out.println("sb1 is " + sb1);
  12. System.out.println("sb2 is " + sb2);
  13. System.out.println("");
  14. System.out.println("calling modify");
  15. //按值传递所有参数
  16. modify(val, sb1, sb2);
  17. System.out.println("returned from modify");
  18. System.out.println("");
  19. System.out.println("val is " + val);
  20. System.out.println("sb1 is " + sb1);
  21. System.out.println("sb2 is " + sb2);
  22. }
  23. public static void modify(int a, StringBuffer r1,
  24. StringBuffer r2)
  25. {
  26. System.out.println("in modify...");
  27. a = 0;
  28. r1 = null;  //1
  29. r2.append(" taste good");
  30. System.out.println("a is " + a);
  31. System.out.println("r1 is " + r1);
  32. System.out.println("r2 is " + r2);
  33. }
  34. }

Java 应用程序的输出

  1. val is 10
  2. sb1 is apples
  3. sb2 is pears
  4. calling modify
  5. in modify...
  6. a is 0
  7. r1 is null
  8. r2 is pears taste good
  9. returned from modify
  10. val is 10
  11. sb1 is apples
  12. sb2 is pears taste good

这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给 modify 方法。
modify 方法更改了所有三个参数的值: 
将第一个参数(整数)设置为 0。  将第一个对象引用 r1 设置为 null。  保留第二个引用 r2 的值,但通过调用 append 方法更改它所引用的对象(这与前面的 C++ 示例中对指针 p 的处理类似)。
当执行返回到 main 时,再次打印出这三个参数的值。正如预期的那样,整型的 val 没有改变。对象引用 sb1 也没有改变。如果 sb1 是按引用传递的,正如许多人声称的那样,它将为 null。但是,因为 Java 编程语言按值传递所有参数,所以是将 sb1 的引用的一个副本传递给了 modify 方法。当 modify 方法在 //1 位置将 r1 设置为 null 时,它只是对 sb1 的引用的一个副本进行了该操作,而不是像 C++ 中那样对原始值进行操作。
另外请注意,第二个对象引用 sb2 打印出的是在 modify 方法中设置的新字符串。即使 modify 中的变量 r2 只是引用 sb2 的一个副本,但它们指向同一个对象。因此,对复制的引用所调用的方法更改的是同一个对象。

传值---传递基本数据类型参数 public    class           PassValue{     static void exchange(int a, int b){//静态方法,交换a,b的值         int temp;         temp = a;         a = b;         b = temp;     }     public static void main(String[] args){        int i = 10;        int j = 100;        System.out.println("before call: " + "i=" + i + "\t" + "j = " + j);//调用前         exchange(i, j);                                                                    //值传递,main方法只能调用静态方法         System.out.println("after call: " + "i=" + i + "\t" + "j = " + j);//调用后     } } 运行结果:         before call: i = 10        j = 100         after    call: i = 10        j = 100 说明:调用exchange(i, j)时,实际参数i,j分别把值传递给相应的形式参数a,b,在执行方法exchange()时,形式参数a,b的值的改变不影响实际参数i和j的值,i和j的值在调用前后并没改变。 引用传递---对象作为参数 如果在方法中把对象(或数组)作为参数,方法调用时,参数传递的是对象的引用(地址),即在方法调用时,实际参数把对对象的引用(地址)传递给形式参数。这是实际参数与形式参数指向同一个地址,即同一个对象(数组),方法执行时,对形式参数的改变实际上就是对实际参数的改变,这个结果在调用结束后被保留了下来。 class Book{     String name;     private folat price;     Book(String n,    float ){                //构造方法         name = n;         price = p;     }     static  void  change(Book a_book,    String n,    float p){    //静态方法,对象作为参数             a_book.name = n;             a_book.price = p;     }     public void output(){        //实例方法,输出对象信息         System.out.println("name: " + name + "\t" + "price: " + price);     } }  public class PassAddr{     public static void main(String [] args){         Book b = new Book("java2",    32.5f);         System.out.print("before call:\t");        //调用前         b.output();         b.change(b,    "c++",    45.5f);            //引用传递,传递对象b的引用,修改对象b的值         System.out.print("after call:\t");            //调用后         b.output();     } } 运行结果:         before    call:    name:java2        price:32.5         after       call:    name:c++          price:45.5 说明:调用change(b,"c++",45.5f)时,对象b作为实际参数,把引用传递给相应的形式参数a_book,实际上a_book也指向同一个对象,即该对象有两个引用名:b和a_book。在执行方法change()时,对形式参数a_book操作就是对实际参数b的操作。

时间: 2024-10-29 10:46:08

Java 中的值传递和参数传递的相关文章

一道笔试题来理顺Java中的值传递和引用传递

前段时间参加了一场面试,其中有一道引用传递的题,因为当时并没有考虑清楚所以做错了. 现在来复盘一下,题目如下: private static void change(StringBuffer str11, StringBuffer str12) { str12 = str11; str11 = new StringBuffer("new world"); str12.append("new world");} public static void main(Stri

为什么说Java中只有值传递----说服自己

在开始深入讲解之前,有必要纠正一下大家以前的那些错误看法了.如果你有以下想法,那么你有必要好好阅读本文. 错误理解一:值传递和引用传递,区分的条件是传递的内容,如果是个值,就是值传递.如果是个引用,就是引用传递. 错误理解二:Java是引用传递. 错误理解三:传递的参数如果是普通类型,那就是值传递,如果是对象,那就是引用传递. 实参与形参 我们都知道,在Java中定义方法的时候是可以定义参数的.比如Java中的main方法: public static void main(String[] ar

为什么大家都说Java中只有值传递?

最近跟Java中的值传递和引用传递杠上了,一度怀疑人生.查了很多资料,加上自己的理解,终于搞清楚了,什么是值传递和引用传递.也搞明白了,为什么大家都说Java只有值传递,没有引用传递.原来,我一直以来的认知都是错误的... 首先,需要了解一些概念性的东西. 形参与实参: 形参,是指在定义函数时使用的参数,目的是用于接收调用该函数时传入的参数.简单理解,就是所有函数(即方法)的参数都是形参. 实参,是指调用函数时,传递给函数的参数. public static void main(String[]

Java中的值传递与“引用传递”

首先,Java没有 引用传递 这么一说. Java只有值传递,传递的都是值,基本数据类型传递的是值,引用类型传递的是地址值. 我来理一下这其中犹如米线跟米面绞在一起了,再跟粉丝混在一起的关系. 好的,我们来看第一个例子: public static void main(String[] args) { int a = 1; int b = 4; System.out.println("Before : a=" + a + "," + "b=" +

浅析java中的值传递与引用传递

public class Test { public static void main(String[] args) { String s = new String("aaa"); change(s); System.out.println(s); StringBuilder sb = new StringBuilder("111"); change(sb); System.out.println(sb); } static void change(String s

Java中的值传递

1.先比较下字符串的比较 == 代表全等于 值和地址(存放地址) 全部相等于. equals 值等于== 和 equals的区别 列如下面的 如果name1==name2是等于的 然而name1==name3 结果 不是不等于的,因为生成了新的地址 如果 name1.equals(name3) 这个比较只要求值的比较,输出结果是相等的. String name1="zhangsan"; String name2="zhangsan"; String name3=ne

java中的值传递和引用传递有什么区别呀?

值传递: (形式参数类型是基本数据类型和String):方法调用时,实际参数把它的值传递给对应的形式参数,形式参数只是用实际参数的值初始化自己的存储单元内容,是两个不同的存储单元,所以方法执行中形式参数值的改变不影响实际参数的值. 引用传递: (形式参数类型是引用数据类型参数除去String):也称为传地址.方法调用时,实际参数是对象(或数组),这时实际参数与形式参数指向同一个地址,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,这个结果在方法结束后被保留了下来,所以方法执行中形式参数

JAVA中只有值传递

今天,我在一本面试书上看到了关于java的一个参数传递的问题: 写道 java中对象作为参数传递给一个方法,到底是值传递,还是引用传递? 我毫无疑问的回答:"引用传递!",并且还觉得自己对java的这一特性很是熟悉! 结果发现,我错了! 答案是: 值传递!Java中只有按值传递,没有按引用传递! 回家后我就迫不及待地查询了这个问题,觉得自己对java这么基础的问题都搞错实在太丢人! 综合网上的描述,我大概了解了是怎么回事,现在整理如下,如有不对之处望大神提出! 先来看一个作为程序员都熟

JAVA中的值传递和引用传递问题

这是个老生常谈的问题了,引起过无数争论,但可以说一直没有一个令人满意的回答. 有人总结过: 对象是按引用传递的 Java 应用程序有且仅有的一种参数传递机制,即按值传递 按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本 按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本 简单总结: 对象就是传引用(引用地址的拷贝,实质也是传值) 原始类型就是传值 String等immutable类型因为没有提供自身修改的函数,每次操作都是新生成一个