Java 应用程序中的按值传递语义
原文http://www.cn.ibm.com/developerWorks/java/passbyval/index.shtml
节选理解参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递。写它是为了揭穿普遍存在的一种神话,即认为 Java 应用程序按引用传递参数,以避免因依赖“按引用传递”这一行为而导致的常见编程错误。
对此节选的某些反馈意见认为,我把这一问题搞糊涂了,或者将它完全搞错了。许多不同意我的读者用 C++ 语言作为例子。因此,在此栏目中我将使用 C++ 和 Java 应用程序进一步阐明一些事实。
要点
读完所有的评论以后,问题终于明白了,至少在一个主要问题上产生了混淆。某些评论认为我的节选是错的,因为对象是按引用传递的。对象确实是按引用传递的;
节选与这没有冲突。节选中说所有参数都是按值 -- 另一个参数 -- 传递的。下面的说法是正确的:在 Java
应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java
应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。
C++ 和 Java 应用程序中的参数传递
Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。这是一个重要特性,正如随后的代码示例所示的那样。
在继续讨论之前,定义按值传递和按引用传递这两个术语是重要的。按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果
函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因
此,如果函数修改了该参数,调用代码中的原始值也随之改变。
关于 Java 应用程序中参数传递的某些混淆源于这样一个事实:许多程序员都是从 C++ 编程转向 Java 编程的。C++
既包含非引用类型,又包含引用类型,并分别按值和按引用传递它们。Java 编程语言有基本类型和对象引用;因此,认为 Java 应用程序像 C++
那样对基本类型使用按值传递,而对引用使用按引用传递是符合逻辑的。毕竟您会这么想,如果正在传递一个引用,则它一定是按引用传递的。很容易就会相信这一
点,实际上有一段时间我也相信是这样,但这不正确。
在 C++ 和 Java 应用程序中,当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++
中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java
应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。请注意,调用方法的对象引用和副本都指向同
一个对象。这是一个重要区别。Java 应用程序在传递不同类型的参数时,其作法与 C++ 并无不同。Java
应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。
示例
我们将使用前面的定义和讨论分析一些示例。首先考虑一段 C++ 代码。C++ 语言同时使用按值传递和按引用传递的参数传递机制:
清单 1:C++ 示例 #include
#include
void modify(int a, int *P, int &r);
int main (int argc, char** argv)
{
int val, ref;
int *pint;
val = 10;
ref = 50;
pint = (int*)malloc(sizeof(int));
*pint = 15;
printf("val is %d\n", val);
printf("pint is %d\n", pint);
printf("*pint is %d\n", *pint);
printf("ref is %d\n\n", ref);
printf("calling modify\n");
//按值传递 val 和 pint,按引用传递 ref。
modify(val, pint, ref);
printf("returned from modify\n\n");
printf("val is %d\n", val);
printf("pint is %d\n", pint);
printf("*pint is %d\n", *pint);
printf("ref is %d\n", ref);
return 0;
}
void modify(int a, int *p, int &r)
{
printf("in modify...\n");
a = 0;
*p = 7;
p = 0;
r = 0;
printf("a is %d\n", a);
printf("p is %d\n", p);
printf("r is %d\n", r);
}
这段代码的输出为:
清单 2:C++ 代码的输出 val is 10
pint is 4262128
*pint is 15
ref is 50
calling modify
in modify...
a is 0
p is 0
r is 0
returned from modify
val is 10
pint is 4262128
*pint is 7
ref is 0
这段代码声明了三个变量:两个整型变量和一个指针变量。设置了每个变量的初始值并将其打印出来。同时打印出了指针值及其所指向的值。然后将所有三个变量作
为参数传递给 modify 函数。前两个参数是按值传递的,最后一个参数是按引用传递的。modify
函数的函数原型表明最后一个参数要作为引用传递。回想一下,C++ 按值传递所有参数,引用除外,后者是按引用传递的。
modify 函数更改了所有三个参数的值:
将第一个参数设置为 0。
将第二个参数所指向的值设置为 7,然后将第二个参数设置为 0。
将第三个参数设置为 0。
将新值打印出来,然后函数返回。当执行返回到 main 时,再次打印出这三个参数的值以及指针所指向的值。作为第一个和第二个参数传递的变量不受
modify 函数的影响,因为它们是按值传递的。但指针所指向的值改变了。请注意,与前两个参数不同,作为最后一个参数传递的变量被 modify
函数改变了,因为它是按引用传递的。
现在考虑用 Java 语言编写的类似代码:
清单 3:Java 应用程序 class Test
{
public static void main(String args[])
{
int val;
StringBuffer sb1, sb2;
val = 10;
sb1 = new StringBuffer("apples");
sb2 = new StringBuffer("pears");
System.out.println("val is " + val);
System.out.println("sb1 is " + sb1);
System.out.println("sb2 is " + sb2);
System.out.println("");
System.out.println("calling modify");
//按值传递所有参数
modify(val, sb1, sb2);
System.out.println("returned from modify");
System.out.println("");
System.out.println("val is " + val);
System.out.println("sb1 is " + sb1);
System.out.println("sb2 is " + sb2);
}
public static void modify(int a, StringBuffer r1,
StringBuffer r2)
{
System.out.println("in modify...");
a = 0;
r1 = null; //1
r2.append(" taste good");
System.out.println("a is " + a);
System.out.println("r1 is " + r1);
System.out.println("r2 is " + r2);
}
}
这段代码的输出为:
清单 4:Java 应用程序的输出 val is 10
sb1 is apples
sb2 is pears
calling modify
in modify...
a is 0
r1 is null
r2 is pears taste good
returned from modify
val is 10
sb1 is apples
sb2 is pears taste good
这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给 modify 方法。
modify 方法更改了所有三个参数的值:
将第一个参数(整数)设置为 0。
将第一个对象引用 r1 设置为 null。
保留第二个引用 r2 的值,但通过调用 append 方法更改它所引用的对象(这与前面的 C++ 示例中对指针 p 的处理类似)。
当执行返回到 main 时,再次打印出这三个参数的值。正如预期的那样,整型的 val 没有改变。对象引用 sb1 也没有改变。如果 sb1
是按引用传递的,正如许多人声称的那样,它将为 null。但是,因为 Java 编程语言按值传递所有参数,所以是将 sb1
的引用的一个副本传递给了 modify 方法。当 modify 方法在 //1 位置将 r1 设置为 null 时,它只是对 sb1
的引用的一个副本进行了该操作,而不是像 C++ 中那样对原始值进行操作。
另外请注意,第二个对象引用 sb2 打印出的是在 modify 方法中设置的新字符串。即使 modify 中的变量 r2 只是引用 sb2 的一个副本,但它们指向同一个对象。因此,对复制的引用所调用的方法更改的是同一个对象。
编写一个交换方法
假定我们知道参数是如何传递的,在 C++ 中编写一个交换函数可以用不同的方式完成。使用指针的交换函数类似以下代码,其中指针是按值传递的:
清单 5:使用指针的交换函数 #include
#include
void swap(int *a, int *b);
int main (int argc, char** argv)
{
int val1, val2;
val1 = 10;
val2 = 50;
swap(&val1, &val2);
return 0;
}
void swap(int *a, int *b)
{
int temp = *b;
*b = *a;
*a = temp;
}
使用引用的交换函数类似以下代码,其中引用是按引用传递的:
清单 6:使用引用的交换函数 #include
#include
void swap(int &a, int &b);
int main (int argc, char** argv)
{
int val1, val2;
val1 = 10;
val2 = 50;
swap(val1, val2);
return 0;
}
void swap(int &a, int &b)
{
int temp = b;
b = a;
a = temp;
}
两个 C++ 代码示例都像所希望的那样交换了值。如果 Java 应用程序使用“按引用传递”,则下面的交换方法应像 C++ 示例一样正常工作:
清单 7:Java 交换函数是否像 C++ 中那样按引用传递参数 class Swap
{
public static void main(String args[])
{
Integer a, b;
a = new Integer(10);
b = new Integer(50);
System.out.println("before swap...");
System.out.println("a is " + a);
System.out.println("b is " + b);
swap(a, b);
System.out.println("after swap...");
System.out.println("a is " + a);
System.out.println("b is " + b);
}
public static void swap(Integer a, Integer b)
{
Integer temp = a;
a = b;
b = temp;
}
}
因为 Java 应用程序按值传递所有参数,所以这段代码不会正常工作,其生成的输入如下所示:
清单 8:清单 7 的输出 before swap...
a is 10
b is 50
after swap...
a is 10
b is 50
那么,在 Java 应用程序中如何编写一个方法来交换两个基本类型的值或两个对象引用的值呢?因为 Java 应用程序按值传递所有的参数,所以您不能这样做。要交换值,您必须用在方法调用外部用内联来完成。
结论
我在书中包括该信息的意图并不是作琐细的分析或试图使问题复杂化,而是想警告程序员:在 Java 应用程序中假定“按引用传递”语义是危险的。如果您在 Java 应用程序中假定“按引用传递”语义,您就可能写出类似上面的交换方法,然后疑惑它为什么不正常工作。
我必须承认,在我第一次认识到 Java 应用程序按值传递所有参数时,我也曾表示怀疑。我曾一直假定因为 Java
应用程序有两种类型,所以他们按值传递基本类型而按引用传递引用,就像 C++ 那样。在转向 Java 编程之前我已用 C++
编程好几年了,感觉任何其他事情似乎都不直观。但是,一旦我理解了发生的事情,我就相信 Java 语言按值传递所有参数的方法更加直观。The
Java Programming Language,Second Edition 的作者,Ken Arnold 和 James Gosling
在 2.6.1 节中说得最好:“在 Java 中只有一种参数传递模式 -- 按值传递 -- 这有助于使事情保持简单。”
分享到:
相关推荐
【摘要】:本文主要探讨了如何在J2ME环境中实现Push信息自动启动JAVA移动应用程序的功能,通过MIDP 2.0的Push注册特性,使JAVA应用程序能够接收并处理来自服务器的推送信息。作者Srijeeb Roy提供了一个详细的步骤...
虽然Swing已经成为现代Java应用程序的首选GUI框架,但AWT中的一些核心概念,如事件处理模型和布局管理器,依然被广泛应用。 - **事件处理及监听**:AWT提供了事件处理机制,允许开发者通过监听器来响应用户操作,...
这对于提高Java程序的效率和性能具有重要的意义,尤其是对于需要高性能计算的应用程序而言。此外,这项工作也指明了如何通过形式化方法来增强现代编译器的能力,这不仅对编译器开发者来说是个宝贵的知识,对于追求...
6. **模块系统**:Java 9引入了模块系统(Project Jigsaw),这是一个重大的架构改变,旨在提高Java应用程序的可维护性、安全性和性能。模块化使得依赖关系更加明确,有助于避免类路径问题。 其次,Java虚拟机规范...
7. **新的 Nashorn JavaScript引擎**:Java 8包含了Nashorn JavaScript引擎,允许在Java应用程序中直接运行JavaScript代码,实现了Java与JavaScript之间的互操作性。 8. **类型推断增强**:Java 8提高了编译器的...
- 设计并实现一个简单的计算器应用程序,包括数字和运算符按钮。 - 创建一个学生信息添加窗口,允许用户输入信息并保存至数据库。 - 实现一个具有多个功能的加法计算器,展示GUI的动态更新能力。 - 开发基于表格的...
Java应用程序有两种主要类型:Application(应用程序)和Applet(小程序)。源代码文件(*.java)经过javac编译器生成字节码文件(*.class),然后由java解释器执行。开发过程中还需要设置环境变量,如path用于定位...
对于Java应用,如果JZMQ库依赖于本地库,可能还需要通过`-Djava.library.path`参数指定这些库的位置。 在Java中使用ZeroMQ时,开发者需要了解以下几个核心概念: 1. **Context**:它是ZeroMQ的上下文对象,负责...
`Error`类的对象通常由JVM生成并抛出,表示严重的问题,比如内存溢出,这些问题通常难以恢复且不建议应用程序尝试处理。`Exception`类则表示程序可以预见和处理的异常,它又分为`RuntimeException`和其他检查型异常...
5. **事件回调和监听器**:允许应用程序注册监听器,以便在接收到特定Diameter事件时进行处理,如接收新的请求、成功响应或错误消息。 6. **错误处理**:处理Diameter消息中的错误,生成适当的错误响应。 7. **...
SessionBean在J2EE应用程序中被用来完成一些服务器端的业务操作,例如访问数据库、调用其他EJB组件。EntityBean被用来代表应用系统中用到的数据。 对于客户机,SessionBean是一种非持久性对象,它实现某些在服务器上...
在Java编程语言中,异常管理是一项至关重要的技能,它确保了程序的稳定性和可靠性。...以上就是关于“java异常管理”的一些关键知识点,通过良好的异常处理,开发者可以构建更加健壮和稳定的Java应用程序。
6. OSWorkflow:OSWorkflow是一个可嵌入到Java应用程序中的工作流引擎,支持多种持久化选项,如EJB、Hibernate和JDBC。此外,它还可以与Spring框架集成,增强其在企业级应用中的使用。 7. wfmOpen:wfmOpen是J2EE...
- **数组映射**:规定了如何将Java中的数组映射到IDL中,以支持复杂数据结构的远程传递。 - **异常映射**:说明了如何将Java中的异常映射到IDL中,以便于处理远程调用过程中的错误情况。 #### 五、运行时问题 讨论...
现代JVM中的垃圾回收器能够智能地选择合适的时机执行回收操作,减少对应用程序性能的影响。 #### GC调优 GC调优主要包括调整堆大小、设置合适的年轻代和老年代比例、选择合适的垃圾回收器等。合理配置可以显著提升...
SessionBean在J2EE应用程序中被用来完成一些服务器端的业务操作,例如访问数据库、调用其他EJB组件。EntityBean被用来代表应用系统中用到的数据。 对于客户机,SessionBean是一种非持久性对象,它实现某些在服务器...
本示例“vc++ 与java通讯DEMO”提供了一个具体的案例,演示了如何使用VC++(Microsoft Foundation Classes)的MFC库与Java应用程序进行通信。下面将详细介绍这个DEMO中的关键知识点。 首先,我们要理解MFC。MFC是...