java引用传递和值传递的问题
1、首先明确一下值传递和引用传递的概念:
值传递:表示方法接收的是调用者提供的值。
引用传递:表示方法接收的是调用者提供的变量地址。
在Java中,变量分为以下两类:
①对于基本数据类型变量(int、long、double、float、byte、boolean、char),Java是传值的副本。
②对于一切对象型变量,Java都是传引用的副本。其实传引用副本的实质就是复制指向地址的指针,只不过Java不像C++中有显著的*和&符号。
需要注意的是:String类型也是对象型变量,所以它必然是传引用副本。String类是final类型的,因此不可以继承和修改这个类。不管Java参数的类型是什么,一律传递参数的副本。
下面这段代码中,显而易见,percent的值是不会改变的。
double percent=10;
editPercent(percent);
System.out.println("方法结束后的percent:"+percent);
public static void editPercent(double x){
x=3*x;
System.out.println("方法内的X:"+x);
}
而在以下这段代码中,传递了一个Emplyee对象,通过调用方法,改变了Employee对象的属性。
Employee boss= new Employee("boss", 5000);
x.raiseSalary(200);
System.out.println("方法结束后的Salary:"+x.getSalary());
}
第一段代码,其执行过程如下:
(1)x被初始化为percent的一个拷贝(10)
(2)x被乘3,但是percent不变
(3)方法结束后,参数变量x不再使用。
第二段代码,其执行过程如下:
(1)x被初始化为boss的拷贝,是一个对象的引用
(2)editSalary应用于这个对象的引用,这时x修改了这个对象,这时boss也引用同一个对象,这个对象的salary被提高了200%
(3)方法结束后,参数变量x不再使用,boss继续引用已经被修改的Employee对象。
可以发现,当传递一个对象参数时,java对其的操作时近似引用传递的。这也是引发争论的关键点。事实上,尽管《java核心技术》中阐述java不存在引用传递,依然有许多人认为java中存在引用传递。以下是一个极易引发争论的程序。
public class Example {
String str = new String("good");
char[] ch = { 'a', 'b', 'c' };
public static void main(String args[]) {
Example ex = new Example();
//执行了这一句之后
ex.change(ex.str, ex.ch);
//ex的str属性没变,但是ch属性被修改了
System.out.print(ex.str + " and ");
System.out.print(ex.ch);
}
public void change(String str, char ch[]) {
str = "test ok";
ch[0] = 'g';
}
}
这段程序输出了什么呢?
good and gbc ch被方法修改了而str没有被修改。
很明显,str和ch[]都是对象传递,传递到方法中的是一个引用了对象的拷贝。那么为什么char[]被修改了而String没有被修改呢。
2、总结:
java中参数传递情况如下:
· 一个方法不能修改一个基本数据类型的参数
· 一个方法可以修改一个对象参数的状态
· 一个方法不能实现让对象参数引用一个新对象
1、首先明确一下值传递和引用传递的概念:
值传递:表示方法接收的是调用者提供的值。
引用传递:表示方法接收的是调用者提供的变量地址。
在Java中,变量分为以下两类:
①对于基本数据类型变量(int、long、double、float、byte、boolean、char),Java是传值的副本。
②对于一切对象型变量,Java都是传引用的副本。其实传引用副本的实质就是复制指向地址的指针,只不过Java不像C++中有显著的*和&符号。
需要注意的是:String类型也是对象型变量,所以它必然是传引用副本。String类是final类型的,因此不可以继承和修改这个类。不管Java参数的类型是什么,一律传递参数的副本。
下面这段代码中,显而易见,percent的值是不会改变的。
double percent=10;
editPercent(percent);
System.out.println("方法结束后的percent:"+percent);
public static void editPercent(double x){
x=3*x;
System.out.println("方法内的X:"+x);
}
而在以下这段代码中,传递了一个Emplyee对象,通过调用方法,改变了Employee对象的属性。
Employee boss= new Employee("boss", 5000);
editSalary(boss);
x.raiseSalary(200);
System.out.println("方法结束后的Salary:"+x.getSalary());
}
第一段代码,其执行过程如下:
(1)x被初始化为percent的一个拷贝(10)
(2)x被乘3,但是percent不变
(3)方法结束后,参数变量x不再使用。
第二段代码,其执行过程如下:
(1)x被初始化为boss的拷贝,是一个对象的引用
(2)editSalary应用于这个对象的引用,这时x修改了这个对象,这时boss也引用同一个对象,这个对象的salary被提高了200%
(3)方法结束后,参数变量x不再使用,boss继续引用已经被修改的Employee对象。
可以发现,当传递一个对象参数时,java对其的操作时近似引用传递的。这也是引发争论的关键点。事实上,尽管《java核心技术》中阐述java不存在引用传递,依然有许多人认为java中存在引用传递。以下是一个极易引发争论的程序。
public class Example {
String str = new String("good");
char[] ch = { 'a', 'b', 'c' };
public static void main(String args[]) {
Example ex = new Example();
//执行了这一句之后
ex.change(ex.str, ex.ch);
//ex的str属性没变,但是ch属性被修改了
System.out.print(ex.str + " and ");
System.out.print(ex.ch);
}
public void change(String str, char ch[]) {
str = "test ok";
ch[0] = 'g';
}
}
这段程序输出了什么呢?
good and gbc ch被方法修改了而str没有被修改。
很明显,str和ch[]都是对象传递,传递到方法中的是一个引用了对象的拷贝。那么为什么char[]被修改了而String没有被修改呢。
2、总结:
java中参数传递情况如下:
· 一个方法不能修改一个基本数据类型的参数
· 一个方法可以修改一个对象参数的状态
· 一个方法不能实现让对象参数引用一个新对象