浅谈Java到底是值传递还是引用传递呢

一、前言

最近在看Java核心卷一,也就是这本书:

在这本书里面也看到了这个问题,Java是值传递还是引用传递,这个问题其实也是很有意思的,之前也看到过这个问题,但是只是依稀记得是值传递,而且网上也有在讨论这个问题的。所以就先说结论吧:是值传递。

二、值传递与引用传递

既然讨论是值传递还是引用传递,那肯定是要知道啥是值传递、引用传递的。

值传递:是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
引用传递:是指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

所以我们就可以做个简单的比较:

值传递 引用传递 
将参数复制一份传递过去 将参数的实际地址传递过去
不会影响到实际参数 会影响到实际参数

我们在方法中,传递参数类型有两种:基本数据类型(数字、布尔)以及对象引用这两种,所以我们就从这两种类型进行分析。

三、基本数据类型

以数字int为例:

public class Test {
    public static void main(String[] args) {
        Test test=new Test();
        int i=1;
        test.incr(i);
        System.out.println("main中i的值大小为"+i);
    }
    public void incr(int i){
        i=i+1;
        System.out.println("incr中i为"+i);
    }
}

main方法中的结果要么为1,要么为2,如果还是1的话,那么很大可能就是值传递,我们看下输出结果:

incr中i为2
main中i的值大小为1

可以看到,main方法中的值仍为1,我们来看下这个的过程是怎么样的:

incr方法中虽然对i的值进行了加一操作,但是他只是将值复制了一份,incr方法执行完毕之后,就会被处理掉,并没有改掉原先的值,所以才会在main方法中打印出i还是原先的值。

四、对象引用

基本数据类型其实比较好解释,对象引用其实还是有那么一点迷惑性的,因为有的人可以认为对象引用是引用传递,他可以向方法中传递一个对象,然后在子方法中修改对象的值,就比如下面的这个例子:
例子一:

public class Test {
    public static void main(String[] args) {
        Test test=new Test();
        Student s1=new Student();
        s1.setId(1);
        test.changeId(s1);
        System.out.println("main中的s1id是"+s1.getId());
    }

    public void changeId(Student student){
        student.setId(2);
        System.out.println("changeId中的id为"+student.getId());
    }

}
class Student{
    private int id;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }
}

一个很简单的例子,创建个student对象,只有id这一个字段,将student对象传递到子方法中,执行结果会是什么呢?

changeId中的id为2
main中的s1id是2

可以看到阿,main方法与changeId方法中,他们最后的id都是2,所以有的人就会认为,修改了对象中的值,所以是值传递。

其实我们看上面值传递与引用传递的概念,引用传递是传递的地址,那么我们就假设对象引用是引用传递方式,那么我传递进去两个对象,交换他们的位置是可以改变他们的指向的,接下来我们就来看一下会不会改变:
例子二:

package com.dong.No2;

public class Test {
    public static void main(String[] args) {
        Test test=new Test();
        Student s1=new Student();
        Student s2=new Student();
        s1.setId(1);
        s2.setId(2);
        test.changeId2(s1,s2);
        System.out.println("main中的s1id是"+s1.getId()+","+"s2id是"+s2.getId());
    }

    public void changeId2(Student s1,Student s2){
        Student s3=s1;
        s1=s2;
        s2=s3;
        System.out.println("changeId2中的s1id是"+s1.getId()+",s2id是"+s2.getId());
    }
}
class Student{
    private int id;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }
}

他的执行结果:

changeId2中的s1id是2,s2id是1
main中的s1id是1,s2id是2

我们发现,这个在子方法中,对象的值是改变了,但是main方法中的值还是原先的样子,那这样就不符合引用传递了,因为他其实并没有改变原先的对象。

如果我们以值传递的观点来解释,那么就可以说的通了,我们传入的两个参数s1、s2在传递过去后,会复制一份为s1复制、s2复制,然后在子方法中,是对这两个复制过后的对象进行的操作,执行完之后,这些复制的对象就会被回收,所以就出现了我们在主方法中,看到这两个对象的值是没有改变的。

事实上也是如此,我们可以同样来解释例子一种,为啥传入了对象,但是主方法中的值却改变了。

我们知道对象这些都是在堆中存储的,我们在向方法中传递的,实际上是这个对象在堆中的地址,我们传递的对象,实际上就是传递的对象的地址:

因为s1与s1复制都是指向的ox123456,s1复制改变了值,那么s1看到的值也就发生了改变,即使是s1复制最后被回收,ox123456的改变不会恢复。

五、结论

所以Java中的传递只有值传递而没有引用传递,只不过传递为基本数据类型的话,是复制的数值,而对象类型的话,则是复制的对象存放地址。

到此这篇关于浅谈Java到底是值传递还是引用传递呢的文章就介绍到这了,更多相关Java值传递内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 浅谈从Java中的栈和堆,进而衍生到值传递

    简述Java中的栈和堆,变量和对象的地址存放和绑定机制 初学java的小白,很多人都搞不清楚java中堆和栈的概念,我们都知道计算机只能识别二进制字节码文件,如果分不清楚对象和变量在内存的地址分配,也就是堆和栈的问题,很多问题比如绑定机制.静态方法.实例方法.局部变量的作用域就会搞不清楚. 首先记住结论: 基本数据类型.局部变量.String类型的直接赋值都是存放在栈内存中的,用完就消失. new创建的实例化对象.String类型的构造方法new出来的对象及数组,是存放在堆内存中的,用完之后靠垃

  • 浅谈Java中真的只有值传递么

    回顾值传递和引用传递 关于Java是值传递还是引用传递,网上有不一样的说法. 1.基本类型或基本类型的包装类以及String是值传递,引用类型是引用传递. 2.Java中只有值传递. 关于这个问题应该是存在争议的.根据测试出来的结果和我们自己的经验,以及口口相传或是上学时老师讲的,我们认为是第一种.但第二种说法的呼声也很高,渐渐地我们也认为第2中才是对的.那么下面我们就来分析一下这个问题. 在谈这个问题之前我们先了解值传递和引用传递的概念及现象.我还记得,值传递和引用传递这些概念是大学里学Jav

  • 带你详细了解Java值传递和引用传递

    1.什么是值传递,什么是引用传递? 值传递(pass by value)是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数. 引用传递(pass by reference)是指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数. 2.值传递和引用传递的区别是什么? 3.Java中只有值传递 3.1 纠正一下大家以前的那些错误看法 错误理解一:值传递和引用传递,区分的条件是传递的内容,如果是个值,就是值传

  • 解析Java按值传递还是按引用传递

    1:按值传递是什么 指的是在方法调用时,传递的参数是按值的拷贝传递.示例如下: public class TempTest { private void test1(int a){ //做点事情 } public static void main(String[] args) { TempTest t = new TempTest(); int a = 3; t.test1(a);//这里传递的参数a就是按值传递 } } 按值传递重要特点:传递的是值的拷贝,也就是说传递后就互不相关了. 示例如下

  • 详解Java引用类型的参数也是值传递

    简述 调用方法的时候,有需要传参数的情况.在Java中,参数的类型有基本类型和引用类型两种. 一开始听到一个说法,Java没有引用传递,但是一直没有太多的思考在上面,直到前不久玩数组的时候,突然间发现把数组引用变量作为参数传递到一个方法当中进行操作之后,再去访问原数组,尽然改变了.于是乎,就想到了之前在C++里面学过的引用传递,突然有一种错愕的感觉,就查了一些资料,探究当Java引用类型变量作为参数传递给方法的时候,到底是值传递还是引用传递. 结论:如果将Java引用类型变量作为参数传递给方法,

  • Java引用传递和值传递栈内存与堆内存的指向操作

    值传递: (形式参数类型是基本数据类型):方法调用时,实际参数把它的值传递给对应的形式参数,形式参数只是用实际参数的值初始化自己的存储单元内容,是两个不同的存储单元,所以方法执行中形式参数值的改变不影响实际参数的值. 引用传递: (形式参数类型是引用数据类型参数):也称为传地址.方法调用时,实际参数是对象(或数组),这时实际参数与形式参数指向同一个地址,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,这个结果在方法结束后被保留了下来,所以方法执行中形式参数的改变将会影响实际参数. 现有

  • java通过实例了解值传递和引用传递

    这篇文章主要介绍了java通过实例了解值传递和引用传递,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 一.java中的值传递的问题 指的是在方法调用时,传递的参数是按值的拷贝传递.示例如下: public static void main(String[] args) { int a=1; change(a); System.out.println("交换a后的值:"+a); } private static void change(

  • 浅谈Java到底是值传递还是引用传递呢

    一.前言 最近在看Java核心卷一,也就是这本书: 在这本书里面也看到了这个问题,Java是值传递还是引用传递,这个问题其实也是很有意思的,之前也看到过这个问题,但是只是依稀记得是值传递,而且网上也有在讨论这个问题的.所以就先说结论吧:是值传递. 二.值传递与引用传递 既然讨论是值传递还是引用传递,那肯定是要知道啥是值传递.引用传递的. 值传递:是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数. 引用传递:是指在调用函数时将实际参数的地址直接传

  • 浅谈Java中对类的主动引用和被动引用

    本文研究的主要是Java中类的主动引用和被动引用,具体介绍如下. 主动引用,这里介绍的是主动引用的五种场景 1.遇到new,getstatic,putstatic,invokestatic这4条字节码指令时,类如果没初始化就会被初始化,创建对象,读取或设置静态字段,调用静态方法. 2.反射 3.子类初始化前会先初始化父类 4.包含main方法的类,虚拟机启动时会先初始化该类 5.使用jdk的动态语言支持时(不明) 被动引用: class SuperClass{ static{ syso("sup

  • 浅谈JavaScript 函数参数传递到底是值传递还是引用传递

    在传统的观念里,都认为JavaScript函数传递的是引用传递(也称之为指针传递),也有人认为是值传递和引用传递都具备.那么JS的参数传递到底是怎么回事呢?事实上以下的演示也完全可以用于Java 首先来一个比较简单的,基本类型的传递: function add(num){ num+=10; return num; } num=10; alert(add(num)); aelrt(num); //输出20,10 对于这里的输出20,10,按照JS的官方解释就是在基本类型参数传递的时候,做了一件复制

  • 浅谈Java finally语句到底是在return之前还是之后执行(必看篇)

    网上有很多人探讨Java中异常捕获机制try...catch...finally块中的finally语句是不是一定会被执行?很多人都说不是,当然他们的回答是正确的,经过我试验,至少有两种情况下finally语句是不会被执行的: (1)try语句没有被执行到,如在try语句之前就返回了,这样finally语句就不会执行,这也说明了finally语句被执行的必要而非充分条件是:相应的try语句一定被执行到. (2)在try块中有System.exit(0);这样的语句,System.exit(0);

  • 浅谈Java中的this作为返回值时返回的是什么

    有时会遇到this作为返回值的情况,那么此时返回的到底是什么呢? 返回的是调用this所处方法的那个对象的引用,读起来有点绕口哈,有没有想起小学语文分析句子成份的试题,哈哈. 一点点分析的话,主干是"返回的是引用": 什么引用呢?"那个对象的引用": 哪个对象呢?"调用方法的那个对象": 调用的哪个方法呢?"调用的是this所位于的方法":这样就清楚了. 再总结一下就是,this作为返回值时,返回的是调用某方法的对象的引用,这

  • 浅谈Java之Map 按值排序 (Map sort by value)

    Map是键值对的集合,又叫作字典或关联数组等,是最常见的数据结构之一.在java如何让一个map按value排序呢? 看似简单,但却不容易! 比如,Map中key是String类型,表示一个单词,而value是int型,表示该单词出现的次数,现在我们想要按照单词出现的次数来排序: Map map = new TreeMap(); map.put("me", 1000); map.put("and", 4000); map.put("you", 3

  • 浅谈Java中hashCode的正确求值方法

    本文研究的主要是Java中hashCode的正确求值方法的相关内容,具体如下. 散列表有一项优化,可以将对象的散列码(hashCode)缓存起来,如果散列码不匹配,就不会检查对象的等同性而直接认为成不同的对象.如果散列码(hashCode)相等,才会检测对象是否相等(equals). 如果对象具有相同的散列码(hashCode),他们会被映射到同一个散列桶中.如果散列表中所有对象的散列码(hashCode)都一样,那么该散列表就会退化为链表(linked list),从而大大降低其查询效率. 一

  • 一文秒懂java到底是值传递还是引用传递

    首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语.按值调用(call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址.一个方法可以修改传递引用所对应的变量值,而不能修改传递值调用所对应的变量值. 它用来描述各种程序设计语言(不只是 Java)中方法参数传递方式. Java 程序设计语言总是采用按值调用.也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参

  • 浅谈Django前端后端值传递问题

    前端后端传值问题总结 前端传给后端 通过表单传值 1.通过表单get请求传值 在前端当通过get的方式传值时,表单中的标签的name值将会被当做action的地址的参数 此时,在后端可以通过get请求相应的name值拿到对应的value值 例子: html中: <form action="{% url 'backweb:select_art' %}" method="post"> {% csrf_token %} <section class=&q

随机推荐