热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

为什么Java中只有值传递

这篇文章主要介绍了为什么Java中只有值传递,帮助大家更好的理解和使用Java,感兴趣的朋友可以了解下

参数传递

在我们日常编写代码的过程中,调用函数可能是最常见的操作了。那么,在调用函数时,参数是怎么样传递的呢?

值传递

相信有很多人都是学C语言入门的,刚开始写代码时,用的最多的就是值传递了。

void plus_one(int a){
  a++;
  printf("a: %d", a);
}

int main(){
  int n = 10;
  plus_one(n);
  printf("n:%d", n);
  return 0;
}

这是一个简单的值传递的例子,无需多言,plus_one函数的作用就是将传进来的数加一,然后输出。所谓值传递,就是直接将实参n的值赋给形参a,赋值完成之后,两者再无瓜葛。

因此,上面的代码可以等效为:

int main(){
  int n = 10;
  
  // plus_one start
  int a;
  a = n;
  a++;
  printf("a: %d", a);
  // plus_one end
  
  printf("n:%d", n);
  return 0;
}

可以看到,值传递简单直观,然而,调用函数并不能改变实参n的值。

指针传递

那么,当我们需要改变实参的值的时候,我们就会想到使用指针传递,也就是所谓的地址传递。

void plus_one(int* p){
  *p = *p + 1;
}

int main(){
  int n = 10;
  plus_one(&n);
  printf("The result is %d", n);
  return 0;
}

这里,我们将实参n的地址传入plus_one函数,在函数中,直接对指针p所指向的值,也就是n做操作,自然就可以改变实参n的值了。

实际上,指针传递也是值传递。我们将上面的代码改写:

int main(){
  int n = 10;
  
  // plus_one start
  int* p;
  p = &n;
  *p = *p + 1;
  printf("The result is %d", n);
  // plus_one end
  
  return 0;
}

可以看到,所谓的指针传递,也只不过是将变量n的地址值赋给指针变量p,实际上也是值传递。

所以,可以不负责任的概括为,C语言中只有值传递;

引用传递

指针固然强大,但是由于代码不易读,难以理解等问题,也是广为诟病。C++作为C语言的超大杯,引入了引用传递来简化指针传递的写法。

void plus_one(int& a){
  a++;
}

int main(){
  int n;
  plus_one(n);
  printf("The result is %d", n);
  return 0;
}

C++中,对&运算符进行了重载,实现了引用传递。具体实现为,在调用plus_one函数时,在函数调用栈中存变量n的地址,而不是n的值。因此,plus_one中的变量a就相当于是n的"别名",对a操作时,自然会改变n的值。

可见,引用传递的底层也是赋值操作。

Java中的参数传递

那么,在Java中,究竟是引用传递,还是值传递呢?

Java中变量分为基本变量和对象,我们不妨分别讨论。

基本变量类型

首先,对于int、char等基本类型,Java是使用值传递的,很容易验证。

static void plusOne(int a){
  a++;
  System.out.println("a: " + a);
}

public static void main(String[] args){
  int n = 10;
  plusOne(n);
	System.out.println("n: " + n);
}

显然,与C语言中一样,这里n的值是不会改变的。

对象

public class PassObject {
  public static void main(String[] args) {
    Dog myDog = new Dog("Test");
    foo(myDog);
    System.out.println(myDog.getName());// TestPlus
  }

  public static void foo(Dog dog) {
    dog.setName("TestPlus");
  }
}

class Dog{

  private String name;

  public Dog(String name) {
    this.name = name;
  }

  public void setName(String name) {
    this.name = name;
  }

  public String getName() {
    return name;
  }
}

通过上面的例子可以看到,传入对象的引用时,是可以改变对象的属性变量的。那么Java在传递对象作为参数时,是引用传递吗?

实际上并非如此,Java中,对象的引用,实际上相当于对象的指针。在Java中操作对象,只有通过引用操作这一种途径。某种意义上,Java中是不能直接操作对象的。

也就是说,在上例中传参时,没有对myDog对象实例做任何操作,只是把myDog引用值赋给了foo函数中的本地变量dog。并没有像引用传递一样,传入对象实体,但是只在栈中保存对象引用的操作。所以,Java中传递对象时,也是值传递。

所以,Java中只有值传递。

值得一提

然而,还是会有一些特殊情况,会让人怀疑上述结论。

数组

上面只分析了基本变量类型和对象,数组呢?

实际上,Java中的数组也是一种对象,数组类也是继承自Object类。在将数组作为参数时,也是传递的数组的引用,并没有传递数组的实体。

public static void changeContent(int[] arr) {
  arr[0] = 10;
}

public static void changeRef(int[] arr) {
  arr = new int[2];
  arr[0] = 15;
}

public static void main(String[] args) {
  int [] arr = new int[2];
  arr[0] = 4;
  arr[1] = 5;

  changeContent(arr);
  System.out.println(arr[0]); // 10
  changeRef(arr);
  System.out.println(arr[0]); // 10
}

在上例中可以看到,将传入的数组引用赋给一个新的数组后,这个引用就不能操作之前的数组了。

关于引用,英文是reference,实际上,我自认为,翻译为句柄是更为贴切的,引用就像是一个柄,一个Handler,你可以用它操作实体,但他并不是实体本身。就像手柄可以操控游戏机,但不是游戏机本身,当你将这个手柄连接到另一个游戏机的时候, 它就不能操控之前的游戏机了。

包装类和String

public static void main(String[] args) {
  Integer n = 1;
  plusOne(n);
  System.out.println(n); // 1
}

private static void plusOne(Integer n) {
  n = n + 1;
  System.out.println(n);// 2
}

在这段代码中,n作为Integer类型实例的句柄,却并没有成功改变对象的值,这是为什么呢?

在Integer类中,存对应值的属性是value,其声明如下:

private final int value;

可见,value值是不能改的,那加的操作是怎么实现的呢?

在上述加一的过程中,会重新new一个Integer对象,让后将这个对象赋给引用n。这样以来,之前的对象自然是不会改变的。

实际上,包装类以及String类的值,都是final的,所以在执行+的过程中,都会重新生成一个对象,然后对它赋值。

以上就是为什么Java中只有值传递的详细内容,更多关于Java 值传递的资料请关注其它相关文章!


推荐阅读
  • C语言注释工具及快捷键,删除C语言注释工具的实现思路
    本文介绍了C语言中注释的两种方式以及注释的作用,提供了删除C语言注释的工具实现思路,并分享了C语言中注释的快捷键操作方法。 ... [详细]
  • c语言\n不换行,c语言printf不换行
    本文目录一览:1、C语言不换行输入2、c语言的 ... [详细]
  • 本文介绍了在Windows系统上使用C语言命令行参数启动程序并传递参数的方法,包括接收参数程序的代码和bat文件的编写方法,同时给出了程序运行的结果。 ... [详细]
  • 树莓派语音控制的配置方法和步骤
    本文介绍了在树莓派上实现语音控制的配置方法和步骤。首先感谢博主Eoman的帮助,文章参考了他的内容。树莓派的配置需要通过sudo raspi-config进行,然后使用Eoman的控制方法,即安装wiringPi库并编写控制引脚的脚本。具体的安装步骤和脚本编写方法在文章中详细介绍。 ... [详细]
  • C语言判断正整数能否被整除的程序
    本文介绍了使用C语言编写的判断正整数能否被整除的程序,包括输入一个三位正整数,判断是否能被3整除且至少包含数字3的方法。同时还介绍了使用qsort函数进行快速排序的算法。 ... [详细]
  • 本文介绍了使用Python解析C语言结构体的方法,包括定义基本类型和结构体类型的字典,并提供了一个示例代码,展示了如何解析C语言结构体。 ... [详细]
  • C语言常量与变量的深入理解及其影响
    本文深入讲解了C语言中常量与变量的概念及其深入实质,强调了对常量和变量的理解对于学习指针等后续内容的重要性。详细介绍了常量的分类和特点,以及变量的定义和分类。同时指出了常量和变量在程序中的作用及其对内存空间的影响,类似于const关键字的只读属性。此外,还提及了常量和变量在实际应用中可能出现的问题,如段错误和野指针。 ... [详细]
  • 恶意软件分析的最佳编程语言及其应用
    本文介绍了学习恶意软件分析和逆向工程领域时最适合的编程语言,并重点讨论了Python的优点。Python是一种解释型、多用途的语言,具有可读性高、可快速开发、易于学习的特点。作者分享了在本地恶意软件分析中使用Python的经验,包括快速复制恶意软件组件以更好地理解其工作。此外,作者还提到了Python的跨平台优势,使得在不同操作系统上运行代码变得更加方便。 ... [详细]
  • 全面介绍Windows内存管理机制及C++内存分配实例(四):内存映射文件
    本文旨在全面介绍Windows内存管理机制及C++内存分配实例中的内存映射文件。通过对内存映射文件的使用场合和与虚拟内存的区别进行解析,帮助读者更好地理解操作系统的内存管理机制。同时,本文还提供了相关章节的链接,方便读者深入学习Windows内存管理及C++内存分配实例的其他内容。 ... [详细]
  • 本文介绍了200个经典c语言源代码,包括函数的使用,如sqrt函数、clanguagefunct等。这些源代码可以帮助读者更好地理解c语言的编程方法,并提供了实际应用的示例。 ... [详细]
  • 本文讲述了作者从最初对软件工程的选择迷茫到逐渐喜欢并坚持学习的经历。作者在大学期间通过学习专业课和参与项目开发,不断挑战自己并取得成就感。虽然曾考虑过转专业和复读,但最终决定坚持学习软件工程,并为自己的未来努力奋斗。作者还提到了大学生活与自己最初的预期不同,但对此并没有太多抱怨。 ... [详细]
  • 本文介绍了GTK+中的GObject对象系统,该系统是基于GLib和C语言完成的面向对象的框架,提供了灵活、可扩展且易于映射到其他语言的特性。其中最重要的是GType,它是GLib运行时类型认证和管理系统的基础,通过注册和管理基本数据类型、用户定义对象和界面类型来实现对象的继承。文章详细解释了GObject系统中对象的三个部分:唯一的ID标识、类结构和实例结构。 ... [详细]
  • ejava,刘聪dejava
    本文目录一览:1、什么是Java?2、java ... [详细]
  • 《2017年3月全国计算机等级考试二级C语言上机题库完全版》由会员分享,可在线阅读,更多相关《2017年3月全国计算机等级考试二级C语言上机题库完全版( ... [详细]
  • 说到C语言的语句块,真是一堆血泪史。第一大坑就是优先级。刚工作那会儿,C的书没看几本,自信满满的认为C语言都会了,拿出搞ACM培养的豪情壮志,代码倒是写得爽,却到处留 ... [详细]
author-avatar
借钱买黄瓜
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有