java中參數傳遞方式話題終結實例

java新手入門面臨的一個經典的話題,本文意在終結這個話題,java中有說法:Java裏面參數傳遞都是按值傳遞,怎麼理解這句話?用文字說明恐怕不容易說明白,說明白恐怕也難以想明白。

前提

先明確一下,按值還是按引用的概念,它是來自c++語言,引用不是漢語詞典中的一個詞,而是c++的概念——“&”這個符號還記得吧?

爲什麼有這個話題呢?其一,是對按引用傳遞理解不透徹;其二,諸多java書籍及討論論點並沒有切中要害。��

一句話概括,按值傳參還是按引用傳參,既然是參數傳遞方式,那麼只針對形參和實參,這裏說的是參數本身,不是參數對象的子對象或孫子對象。

有了前提,上c++代碼:

#include <iostream>

using namespace std;

class User
{
    private:
        int m_id;
    public:
        User(int id=0){m_id = id;}
        void setId(int id){m_id = id;}
        int getId(){return m_id;}
};

void test0(User t){//按值傳參
  User s;
  t = s;
  t.setId(1002);
  cout << "test1:" << t.getId() << endl;
}

void test1(User *t){//按值傳參
  t = new User();//指針指向了一個新對象,外面實參沒變
  t->setId(1002);
  cout << "test1:" << t->getId() << endl;
}

void test2(User* & t){//按引用傳參
    t = new User();//指針指向了一個新對象,外面實參也跟着變了
    t->setId(1002);
    cout << "test2:" << t->getId() << endl;
}

int main(int argc, char const *argv[]) {
  cout<< "\npass by ref:"<<endl;
  User* t = new User();
  t->setId(1001);
  cout << t->getId() << endl;
  test2(t);
  cout << t->getId() << endl;

  cout<< "\npass by value:"<<endl;
  t = new User();
  t->setId(1001);
  cout << t->getId() << endl;
  test1(t);
  cout << t->getId() << endl;
  return 0;
}

輸出結果:

pass by ref:
1001
test2:1002
1002

pass by value:
1001
test1:1002
1001

c++小結:

按值傳遞,那麼在函數內修改了形參指向一個新對象,外面的實參不受影響。

按引用傳遞,那麼在函數內修改了形參指向一個新對象,外面的實參也變了。

旨在說明問題,代碼可能有內存泄漏。

上java:

package com.pollyduan.bean;

@Data
public class User {
    private Integer id;

    public static void testObject(User t){
        t=new User();//指向了一個新對象,外面實參沒變
        t.setId(1002);
        System.out.println("testObject="+t);
    }

    @Test
    public void testObject(){
        User user=new User();
        user.setId(1001);
        System.out.println("user="+user);
        testObject(user);
        System.out.println("user="+user);
    }
}

輸出結果:

user=User(id=1001)
testObject=User(id=1002)
user=User(id=1001)

java小結:

跟c++的邏輯比較一下,請自行對號入座。

萬事無絕對,你可能發現jdk中有引用傳參的例子,如:

char[] gg={'a','b','c'};
char[] newGG=new char[gg.length];
System.arraycopy(gg,0,newGG,0,gg.length);

看下源碼,它不是jdk本身的方法,是JNI,也就是說是jvm的JNI庫封裝的方法,知道即可。

發佈了150 篇原創文章 · 獲贊 119 · 訪問量 73萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章