java中参数传递方式详解

网友投稿 248 2023-05-31


java中参数传递方式详解

java中参数传递方式详解

java新手入门面临的一个经典的话题,本文意在终结这个话题,java中有说法:Java里面参数传递都是按值传递,怎么理解这句话?用文字说明恐怕不容易说明白,说明白恐怕也难以想明白。

前提

先明确一下,按值还是按引用的概念,它是来自c++语言,引用不是汉语词典中的一个词,而是c++的概念——“&”这个符号还记得吧?

为什么有这个话题呢?其一,是对按引用传递理解不透彻;其二,诸多java书籍及讨论论点并没有切中要害。

一句话概括,按值传参还是按引用传参,既然是参数传递方式,那么只针对形参和实参,这里说的是参数本身,不是参数对象的子对象或孙子对象。

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

#include

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:"<

User* t = new User();

t->setId(1001);

cout << t->getId() << endl;

test2(t);

cout << t->getId() << endl;

cout<< "\npass by value:"<

t = new User();

iUSqwPkgUt->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.printhttp://ln("user="+user);

}

}

输出结果:

user=User(id=1001)

testObject=User(id=1002)

usiUSqwPkgUer=User(id=1001)

java小结:

跟c++的逻辑比较一下,请自行对号入座。

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:"<

User* t = new User();

t->setId(1001);

cout << t->getId() << endl;

test2(t);

cout << t->getId() << endl;

cout<< "\npass by value:"<

t = new User();

iUSqwPkgUt->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.printhttp://ln("user="+user);

}

}

输出结果:

user=User(id=1001)

testObject=User(id=1002)

usiUSqwPkgUer=User(id=1001)

java小结:

跟c++的逻辑比较一下,请自行对号入座。

User* t = new User();

t->setId(1001);

cout << t->getId() << endl;

test2(t);

cout << t->getId() << endl;

cout<< "\npass by value:"<

t = new User();

iUSqwPkgUt->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.printhttp://ln("user="+user);

}

}

输出结果:

user=User(id=1001)

testObject=User(id=1002)

usiUSqwPkgUer=User(id=1001)

java小结:

跟c++的逻辑比较一下,请自行对号入座。

t = new User();

iUSqwPkgUt->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.printhttp://ln("user="+user);

}

}

输出结果:

user=User(id=1001)

testObject=User(id=1002)

usiUSqwPkgUer=User(id=1001)

java小结:

跟c++的逻辑比较一下,请自行对号入座。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:canvas实现环形进度条效果
下一篇:Java ThreadPoolExecutor的参数深入理解
相关文章

 发表评论

暂时没有评论,来抢沙发吧~