本问题已经有最佳答案,请猛点这里访问。
我的变量范围有问题。
public static void main(String[] args){
int[] test={1,2,3};
test(test);
System.out.println(test[0]+""+test[1]+""+test[2]);
}
static void test(int[] test){
test[0]=5;
}
我期望输出为1 2 3,但结果为5 2 3。
为什么我在方法中更改了数组中的值,但原始数组发生了变化?
因为你正在修改对象的状态......
不要对参数的名称感到困惑。在test(..)方法中将参数名称更改为whatever,它不会影响任何内容。
@SotiriosDelimanolis我真的没有在这个问题上得到你的意见=
@LuiggiMendoza OP陈述Why I changed the value in the array in the method, but the original array changed?我的印象是他们认为数组被复制或其他什么。
OP:这与范围无关。您将对象引用值传递给方法,并通过方法内的引用修改了对象。这就是Java(以及C和......)的工作原理。请参阅:有关该主题的Oracle教程
@SotiriosDelimanolis将通过值传递,如果你考虑它(我的意思是,考虑只传递数组的值),这是正确的。但您改为传递引用的副本,因此如果更改对象的状态(在本例中为数组中的条目),则原始对象也将被修改。
@LuiggiMendoza对,如果这是他们的误解,那么这是重复的。他们似乎也对变量的范围感到困惑,可能认为main中的test与test(..)中的test有某种关系。
那么,我该怎么做才能保留原始数组,只需在方法中使用数组的副本?即使我在方法中创建了一个像'int [] temp = test'这样的新数组,也会出现同样的情况。
@ user3288447因为那不会复制数组。我强烈推荐阅读这些教程,或者是一本关于java的好初学者的书。
使用Arrays#copyOf(yourArrayVar, yourArrayVar.length);
int[] temp=test不会创建新的数组对象,它只是将数组引用(即指针)从"test"复制到"temp"。
Java中的数组是一个对象。当您通过new创建数组时,它会在堆上创建,并返回一个引用值(类似于C中的指针)并将其分配给您的变量。
在C中,这将表示为:
int *array = malloc(10 * sizeof(int));
将该变量传递给方法时,您将传递的参考值(已复制)传递给方法中的本地(堆栈)变量。不复制数组的内容,只复制参考值。再次,就像将指针传递给C中的函数一样。
因此,当您通过该引用修改方法中的数组时,您将修改堆上存在的单个数组对象。
您评论说您通过int[] temp=test再次对数组进行了"复制"...这只会生成指向内存中单个数组的引用值(指针)的副本。您现在有三个变量都对同一个数组保持相同的引用(一个在main()中,两个在您的方法中)。
如果要复制数组的内容,Java会在Arrays类中提供静态方法:
int[] newArray = Arrays.copyOf(test, test.length);
这将在堆上分配一个新的数组对象(由第二个参数指定的大小),将现有数组的内容复制到它,然后返回对该新数组的引用。
假设OP的test方法包含test = new int[] {7, 7, 7},实质上是创建一个新数组并将指针重定向到该数组。 为什么这次,外面的阵列没有改变?
定义:
Reference =一个变量,指向数组所在的内存中的位置。
Reference的值=实际的内存地址位置本身
您将数组引用的值传递给test()方法。由于java是Pass By Value,它传递引用的值,而不是数组的值(即副本)。
如果您有C背景,可能更容易将引用视为指针。所以引用的值本质上就是它的内存地址(我在这里捏造java规则,但这可能是最简单的想法)
因此,在您的示例中,将指向数组的引用值传递给test()方法,然后使用该引用值查找数组在内存中的位置,以便它可以访问数组中的数据。
因为在你的test()方法中你没有改变你的数组的引用(它指向的位置,即test = new int[10];),那么你的test()方法会对数组中的原始数据起作用(因为它仍然指向你的原始数组) location),导致元素0被设置为值5。
每当有人说通过价值传递"令人困惑"时,我就完全糊涂了。 C按值传递指针。 Java完全一样。
通过混淆更多?