动态网站制作指南 [  QQ表情  ]
[ 投票调查 ]
[ 企业邮箱 ]
[ 网站空间 ]
网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
ASP源码 | .Net源码 | PHP源码 | JSP源码 | JAVA源码 | CGI源码 | VB源码 | C++源码 | Delphi源码 | PB源码 | VF源码 | 汇编 | 服务器
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > Java教程
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
文章搜索服务
邮件订阅
输入你的邮件地址,
你将不会错过任何关于:
[ Java教程 ]的信息

本月文章推荐
.JAVA中用动态代理类实现记忆功能.
.在J2EE Web 应用中使用.
.RMI规范--第六章.
.如何实例化一个抽象类.
.2005年4月8日--正式开始我的Java.
.Java框架研究——JSF与Struts的异.
.按位“或”赋值运算符 (|=).
.轻松测试-学习如何简化测试外部资.
.主类型的过载.
.Java对象持久化技术之Hibernate入.
.Java 12大特色介绍.
.全面解析定义方法.
.玩转Java的CLASSPATH(四)总结.
.Java 5.0 多线程编程实践.
.java开源数据库项目Derby的问答.
.java与c/c++进行socket通信的一些.
.如何实现将统计表格保存成excel文.
.对等计算实践:P2P遇上SSL.
.我看多态——多态的学习笔记.
.JBuilder2005实现重构之类内部提.

Java应用程序中的按值传递语义

发表日期:2008-1-5 |



  几个月前,developerWorks 发布了我的书 Practical Java 中的一些节选,该书是由 Addison-Wesley 出版的。首先我将利用 developerWorks 上的此栏目回答读者提出的一些问题,然后对有关这些节选的各种评论作一答复。
  
  
  
  节选理解参数是按值而不是按引用传递的说明 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
上一篇:用javaapi进行sort 人气:577
下一篇:透过Java参数来改善Java效能 人气:499
浏览全部Java的内容 Dreamweaver插件下载 常用网页广告代码全集
  最新网站源码 最新软件下载
2008-10-12 team论坛 v2.0.4 bulid 080916 A
2008-10-12 Roclog v3.1.6
2008-10-12 SupeV v1.0.1 简体中文 GBK
2008-10-12 NetCMS v1.6.0.1010 正式版
2008-10-12 PHP考试系统PPFrame v1.2.7
2008-10-12 LPAS个人相册 v1.6.3
2008-10-12 快问仿百度知道系统 动态-静态-互
2008-10-12 方卡广告防点击系统 V1.0 GB2312
2008-10-12 泡菜内容管理系统[PCMS] v1.0 Bu
2008-10-11 联系人分组工具 v1.1 中文破解版
2008-10-11 FaceMelter变脸 v2.0 汉化破解版
2008-10-11 PathTracker道路跟踪仪 v1.2 破解
2008-10-11 Rooms手机聊天室 v0.6.7 破解版
2008-10-11 RemoteDesktop远程桌面 v1.0 破解
2008-10-11 ProRemote远程调音台 v1.0.1 破解
2008-10-11 PicShare照片共享 v1.0.0 破解版
2008-10-11 Photogene照片编辑器 v1.5 汉化破
2008-10-11 WriteRoom共享文档 v1.0 破解版
  发表评论
姓 名: 验证码:
内 容:
站长工具:网站收录查询 | Google PR查询 | ALEXA排名查询 | CSS在线编辑器 | 广告代码 | js/vbs加密 | md5加密 | 进制转换 | UTF-8 转换工具 | Html转换js | Html转换asp | Html转换php | Html转换perl
实用工具:汉字翻译拼音 | 拼音字典 | 符号对照表 | 个税计算 | 实时汇率查询换算 | 经典小工具 | 汉字简繁转换 | 普通单位换算 | 公制单位换算 | 生辰老黄历 | 国内电话区号 | 国家代码与域名缩写 | 文字加密解密 | 健康查询 | 万年历 | 汉字横竖排版 | 手机号码查询 | 计算器 | ip搜索
业务联系 | 广告刊登 | 频道合作 | 投稿荐稿 | 联系方式 | 加入收藏 | RSS订阅
Copyright © 2000-2008 www.knowsky.com All rights reserved | 网络实名:动态网站制作指南 | 沪ICP备05001343号