3年前 (2022-04-27)  Java系列 |   抢沙发  454 
文章评分 0 次,平均分 0.0

为什么按值调用和按引用调用?

方法或函数可以通过两种方式调用。一种是按值调用,另一种是按引用调用,这两种方式通常根据作为输入或参数传递给它们的值的类型来区分。

在我们开始之前,让我澄清一件事,在Java中,只有按值调用(值传递),没有按引用调用

按值调用

按值调用是一种方法,通过复制给定变量(或常量或任何保存数据的对象)的实际值,将参数传递给函数

这意味着当我们调用一个传递参数值的方法时,这个参数值会被复制到内存的一部分(取决于类型),值的副本会被传递给调用方法的参数。

在Java中,只有按值调用,没有按引用调用。

引用调用

通过引用调用是一种将值的引用(即地址)传递给方法的方法。

这意味着,当我们通过将引用的副本(即地址)传递给该方法的值而不是值本身的副本来调用该方法时,调用方法具有该值的引用(即地址)。

当调用方法更改值的引用时,原始引用也会更改。

为什么Java中只有按值调用,没有按引用调用?

根据定义,引用调用是我们传递该变量的引用(即地址)的地方。要做到这一点,我们需要首先存储地址,这将需要一个指针(在C中,其传递为*variableName)。正如我们所知,在java中,我们没有指针,因此不可能传递引用(即地址)并执行逐引用调用。

Java不支持指针,这就是为什么Java不支持引用调用。但是C/C++支持指针,因此这些语言支持引用调用。

您可能会问,为什么Java不支持指针?好吧,这是另一个有争议的话题,我们可以在其他文章中讨论,直到签出这个stackoverflow页面(https://stackoverflow.com/questions/5298421/why-doesnt-java-support-pass-by-reference-like-c),这让您对它有了一些了解。

但是,我们仍然可以借助对象行为在Java中实现引用调用。

Java中的引用调用,但如何调用?

Java总是使用按值调用。这意味着该方法获取所有参数值的副本,并且该方法不能修改传递的变量的内容。Java使用两种方法参数:

  • Java基本类型
  • Java对象引用

它看起来非常简单明了,就像将基本类型传递给方法一样。但在将对象传递给该方法时,这是非常引人注目的。当一个对象被传递给一个方法时,该方法会获得对象引用的一个副本,而原始副本和正式副本都引用同一个对象,因此在调用方法中,可以更改对象参数的状态。

让我们通过一个例子来详细了解上述两个方法参数。

将Java基本数据类型作为参数传递给方法

对于原始数据类型参数,原始值不变。让我们举一个简单的例子:

public class PrimitiveDataInAction {
  public static void main(String[] args) {

    int someValue = 50;

    System.out.println("Before method call :: " + someValue);
    
    modified(someValue);
    
    System.out.println("After method call :: " + someValue);
  }

  public static void modified(int someValue) {
    System.out.println("exiting name :: " + someValue);
    // modifying the parameter
    someValue = 15;

    System.out.println("modified name :: " + someValue);
  }
}

输出:

Before method call :: 50
exiting name :: 50
modified name :: 15
After method call :: 50

Process finished with exit code 0

当您将原始数据类型变量传递给一个方法时,它会为该方法创建一个本地副本,并执行方法中提到的任何操作&一旦这个变量从这个方法中出来,方法本地副本就会从内存中删除

&调用该变量之前加载的前一个值。

Java对象引用

对于Java对象引用类型的按值调用,会更改原始值。让我们举一个简单的例子:

class JavaObjectReferencesInAction {
  public static void main(String[] args) {

    List <String> listOfString = new ArrayList < > ();
    listOfString.add("Mahesh");

    System.out.println("Before modified :: " + listOfString);

    modified(listOfString);

    System.out.println("After modified :: " + listOfString);
  }

  public static void modified(List < String > listOfString) {
    // modifying the string
    listOfString.add("Imran");
  }
}

输出:

Before modified :: [Mahesh]
exiting listOfString :: [Mahesh]
modified listOfString :: [Mahesh, Imran]
After modified :: [Mahesh, Imran]

Process finished with exit code 0

让我们看看它是如何工作的。

在上面的示例中,我们创建了一个ArrayList对象,并在其中添加了一个字符串值Mahesh。现在列表中的数据是一个字符串,值为Mahesh

在下一行中,我们调用了modified方法,该方法获取一个列表,并向同一给定列表中添加一个字符串Imran

但是当我们调用这个修改过的方法时,Java对象引用代替了值。它传递list的引用,list只是list的实际对象

这也会导致修改原始列表。

结论

我们详细讨论了一个棘手的面试问题,即Java中的按值调用和按引用调用。对于初学者来说,这可能需要消化很多东西,但相信我,一旦你开始玩它,它就会在你的脑海中清晰地浮现出来。

  
 

除特别注明外,本站所有文章均为老K的Java博客原创,转载请注明出处来自https://javakk.com/2626.html

关于

发表评论

表情 格式

暂无评论

登录

忘记密码 ?

切换登录

注册