`
run_wang
  • 浏览: 166737 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
文章分类
社区版块
存档分类
最新评论

JAVA中的引用到底是传值还是传址?

阅读更多
JAVA中除了8种基本类型外,其它的类型是引用类型,像STRING,数组,文件流等。引用变量在JAVA中是一个存储对象在内存中的地址的变量。所以字符串内容的比较不是直接用等号,而是用字符串的方法equeals()来比较内容的。
1. 简单类型是按值传递的

 Java 方法的参数是简单类型的时候,是按值传递的 (pass by value)。这一点我们可以通过一个简单的例子来说明:

     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 来举一个例子:

    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 对象。

      这里我描述了两个要点:

    1. 引用是一种数据类型,保存了对象在内存中的地址,这种类型即不是我们平时所说的简单数据类型也不是类实例(对象);
    2. 不同的引用可能指向同一个对象,换句话说,一个对象可以有多个引用,即该类类型的变量。


3. 对象是如何传递的呢

  关于对象的传递,有两种说法,即“它是按值传递的”和“它是按引用传递的”。这两种说法各有各的道理,但是它们都没有从本质上去分析,即致于产生了争论。

  既然现在我们已经知道了引用是什么东西,那么现在不妨来分析一下对象作是参数是如何传递的。还是先以一个程序为例:

    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 是一个引用,这一点是勿庸置疑的。前面提到,引用是一种数据类型,而且不是对象,所以它不可能按引用传递,所以它是按值传递的,它么它的值究竟是什么呢?是对象的地址。

  由此可见,对象作为参数的时候是按值传递的,对吗?错!为什么错,让我们看另一个例子:

    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 语言中有一种数据类型叫做指针,于是将一个数据作为参数传递给某个函数的时候,就有两种方式:传值,或是传指针,它们的区别,可以用一个简单的例子说明:

    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,我想写一个方法来交换它们的值,应该怎么办?

  结论很让人失望——没有办法!因此,我们只能具体情况具体讨论,以经常使用交换方法的排序为例:

    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 所指示的对象的成员数据,

    即上述讨论的第二种改变参数值的方法。

    希望大家能够举一反三,使用类似的方法来解决相关问题。

分享到:
评论

相关推荐

    Java是传值还是传址引用

    ### Java是传值还是传址引用 #### 一、简单类型是按值传递的 Java在处理简单数据类型(如int、boolean等)时采用的是按值传递的方式。这意味着当你将一个简单类型的值作为参数传递给一个方法时,实际上传递的是这...

    javascript中的变量是传值还是传址的?

    传址和传引用是一回事。 一门编程语言的核心是数据结构,粗略来讲,可以把数据结构分成不可变类型(immutable)和可变类型(mutable)。为什么这么分呢?这涉及到内存分配问题。对于不可变类型,只要分配有限的内存...

    javascript的变量、传值、传址、参数之间关系

    先把收获晾一下: 1.javascrip变量包含两种类型的值,一种为引用类型的值,一种是基本类型的值。引用类型包括:Array,...”当一个变量复制引用类型的值时,同样也会将存储在变量对象中的值复制一份放到为新变量分配的

    java的类别方法(格式:PPT 字体:繁体)

    Java中的参数传递总是按值传递,意味着方法内部不能直接修改传入参数的原始值,除非参数是对象引用,此时传递的是对象引用的副本,而不是实际对象。 4-6 递归程序设计 递归是解决复杂问题的一种技术,通过将大问题...

    Java初级教材 连载

    传值和传址是两种不同的操作方式,传值复制的是值,而传址复制的是对象的引用。基本数据类型通常通过传值操作,而对象数据类型(如数组和函数)通过传址操作。 3. **字符串的特殊性**: JavaScript中的字符串是不...

    java判断题题库.doc

    Java中的方法调用遵循传值的原则,即使是对象也是传递对象的引用。 51. **非静态方法不能引用静态变量** 错误。非静态方法可以访问静态变量。 52. **静态初始化器是在构造方法被自动调用之前运行的** 正确。...

    2021-2022计算机二级等级考试试题及答案No.19095.docx

    - 实参传递给形参时,可以是传值或传址,传址情况下,形参是实参的副本,但它们指向同一内存地址。 13. ASP.NET发布网站: - `App-Code` 文件夹在发布后通常不会存在,因为编译时代码会被合并到输出目录的其他...

    java 面试题

    还讨论了参数传递的方式,包括基本类型的传值与引用类型的传址,以及final参数的特殊性。同时,介绍了不同类型之间的转换,包括原始类型转换、String类转换和日期类型转换。 声明和访问控制章节重点讲解了数组的...

    2021-2022计算机二级等级考试试题及答案No.1494.docx

    - 函数调用时参数传递可以是传值或传址。 - 函数可以返回任何类型的值,包括结构体。 以上知识点涵盖了数据库备份、Java GUI编程、网络通信、网页开发、电子表格操作、内存管理、计算机制作、数据模型、文件处理...

    JavaScript 面试基础

    JavaScript 中变量的作用域相对与 JAVA、C 这类语言显得更自由,一个很大的特征就是 JavaScript 变量没有块级作用域,函数中的变量在整个函数都中有效。 例如: ```javascript function outPut(s) { document....

    EssentialC

    它们可以接受参数并返回值,分为值参数(传值调用)和引用参数(传址调用)。值参数传递的是参数的副本,不会改变原值;引用参数则传递的是参数的地址,可以修改原值。此外,C语言还支持const关键字,用于声明不可变...

    活动1

    3. **函数**:C语言使用函数组织代码,学习如何定义、调用函数,以及函数参数的传递方式(传值和传址)。 4. **数组和字符串**:数组是一组相同类型的数据集合,字符串是字符数组的特殊形式。了解如何操作数组和...

    2023软考-2023上半年冲刺软件设计师知识点.pdf

    5. **传值调用和传址调用**:函数调用时参数传递的不同方式,前者复制实参值,后者传递实参地址。 6. **翻译阶段**:编译器经过词法分析、语法分析、语义分析后生成目标代码。 7. **符号表**:记录源代码中变量、...

Global site tag (gtag.js) - Google Analytics