说说Java到底是值传递仍是引用传递

时间:2023-05-16

说句真实话,离读者trustyou发给我这段信息现已过去1周时刻了。不是我怠慢,确实是可更新的内容真实是太多了。这不,又有两个读者不约而同地要求我更新一下Java究竟是值传递仍是引证传递方面的文章——其实这个问题我之前是写过的,但现在看起来答案好像不够尽善尽美,所以打算以面试的角度重写一篇。
七年前,我从温和湿润的苏州回到古色古香的洛阳,抱着一幅“全国我有”的心态“约谈”了几位面试官。其中有一位叫老马,让我印象深刻。由于他其时扔了一个面试题把我砸懵了:说说Java究竟是值传递仍是引证传递吧。
我其时年轻气盛,自认为所有的面试题都能对答如流,没想到被老马“刁难”了——本来洛阳这块互联网的荒漠也有技术专家啊。现在回想起来,脸上不自觉地就泛起了惭愧的红晕:其时真菜!不管怎么说,是时分写篇文章剖析一下值传递仍是引证传递的差异了。
将参数传递给办法有两种常见的办法,一种是“值传递”,一种是“引证传递”。C语言本身只支撑值传递,它的衍生品C++既支撑值传递,也支撑引证传递,而Java只支撑值传递。
01、值传递VS引证传递
首先,咱们必需求搞清楚,究竟什么是值传递,什么是引证传递,不然,讨论Java究竟是值传递仍是引证传递就显得毫无意义。
当一个参数按照值的办法在两个办法之间传递时,调用者和被调用者其实是用的两个不同的变量——被调用者中的变量(原始值)是调用者中变量的一份拷贝,对它们当中的任何一个变量修正都不会影响到另外一个变量。
而当一个参数按照引证传递的办法在两个办法之间传递时,调用者和被调用者其实用的是同一个变量,当该变量被修正时,两边都是可见的。
Java程序员之所以容易搞混值传递和引证传递,主要是由于Java有两种数据类型,一种是根本类型,比如说int,另外一种是引证类型,比如说String。
根本类型的变量存储的都是实际的值,而引证类型的变量存储的是目标的引证——指向了目标在内存中的地址。值和引证存储在stack(栈)中,而目标存储在heap(堆)中。
之所以有这个差异,是由于:
栈的优势是,存取速度比堆要快,仅次于直接坐落CPU中的寄存器。但缺点是,栈中的数据大小与生计周期必须是确认的。
堆的优势是可以动态地分配内存大小,生计周期也不必事前告知编译器,Java的废物回收器会主动收走那些不再运用的数据。但由于要在运行时动态分配内存,存取速度较慢。Java
02、根本类型的参数传递
众所周知,Java有8种根本数据类型,分别是int、long、byte、short、float、double、char和boolean。它们的值直接存储在栈中,每逢作为参数传递时,都会将原始值(实参)复制一份新的出来,给形参用。形参将会在被调用办法结束时从栈中铲除。
来看下面这段代码:
publicclassPrimitiveTypeDemo{
publicstaticvoidmain(String[]args){
intage=18;
modify(age);
System.out.println(age);
}
privatestaticvoidmodify(intage1){
age1=30;
}
}
1)main办法中的age是根本类型,所以它的值18直接存储在栈中。
2)调用modify()办法的时分,将为实参age创立一个副本(形参age1),它的值也为18,不过是在栈中的其他位置。
3)对形参age的任何修正都只会影响它自身而不会影响实参。
03、引证类型的参数传递
来看一段创立引证类型变量的代码:
Writerwriter=newWriter(18,”缄默沉静王二”);
writer是目标吗?仍是目标的引证?为了搞清楚这个问题,咱们可以把上面的代码拆分为两行代码:
Writerwriter;
writer=newWriter(18,”缄默沉静王二”);
假设writer是目标的话,就不需求经过new关键字创立目标了,对吧?那也就是说,writer并不是目标,在“=”操作符执行之前,它仅仅是一个变量。那谁是目标呢?newWriter(18,”缄默沉静王二”),它是目标,存储于堆中;然后,“=”操作符将目标的引证赋值给了writer变量,于是writer此刻应该叫目标引证,它存储在栈中,保存了目标在堆中的地址。
每逢引证类型作为参数传递时,都会创立一个目标引证(实参)的副本(形参),该形参保存的地址和实参一样。
来看下面这段代码:
publicclassReferenceTypeDemo{
publicstaticvoidmain(String[]args){
Writera=newWriter(18);
Writerb=newWriter(18);
modify(a,b);
System.out.println(a.getAge());
System.out.println(b.getAge());
}
privatestaticvoidmodify(Writera1,Writerb1){
a1.setAge(30);
b1=newWriter(18);
b1.setAge(30);
}
}
1)在调用modify()办法之前,实参a和b指向的目标是不一样的,尽管age都为18。
2)在调用modify()办法时,实参a和b都在栈中创立了一个新的副本,分别是a1和b1,但指向的目标是一致的(a和a1指向目标a,b和b1指向目标b)。
3)在modify()办法中,修正了形参a1的age为30,意味着目标a的age从18变成了30,而实参a指向的也是目标a,所以a的age也变成了30;形参b1指向了一个新的目标,随后b1的age被修正为30。
修正a1的age,意味着同时修正了a的age,由于它们指向的目标是一个;修正b1的age,对b却没有影响,由于它们指向的目标是两个。
程序输出的成果如下所示:
果然和咱们的剖析是符合的。
04、最后
好了,我亲爱的读者朋友,以上就是本文的全部内容了。看完之后,再遇到面试官问Java究竟是值传递仍是引证传递时,就不必担心被刁难了。我是缄默沉静王二,一枚有趣的程序员。原创不易,莫要白票,请你为本文点赞个吧,这将是我写作更多优质文章的最强动力。

文章标签:

Copyright © 2016 广州思洋文化传播有限公司,保留所有权利。 粤ICP备09033321号

与项目经理交流
扫描二维码
与项目经理交流
扫描二维码
与项目经理交流
ciya68