java参数传递

来源:互联网 发布:数据线材厂家 编辑:程序博客网 时间:2024/06/07 15:29

1.前言讨论


java程序只有传值,没有传引用,传地址的说法。但是传递的值可以是具体的数值,也可以是一个对象的引用。可以用这样一句话来描述“java函数是传值的,java函数传递的参数是对象的引用”。

所以举例子之前,先从jvm的实现原理上有个了解应当是不无裨益的。jvm的结构图可以从《深入java虚拟机》这本巨牛的书上找到,绝对有权威性。从jvm的结构图上可以看出来,jvm在实现的时候将属于它的内存分为五部分,其中程序代码(严格的说应当是字节码)是放在java栈的栈帧中,而对象是从堆中分配的,堆这个东西我看可以理解成“对象池”。程序和程序中需要用到的对象放在两个相对独立的区域中,那么程序怎么使用对象呢?答案是程序中真正使用对象的地方其实只是声明了一个对象的引用,也就是把堆中分配了的相应对象的地址放到引用中,栈和堆之间就是通过一个一个的引用来联系的。引用嘛,我理解就是一个指针常量,指针常量又是个什么东西呢?说白了,就是一个无符号整数,这个整数所表达的是引用对象的地址。好了,这下清楚了,不管是基本类型变量(int,float,double什么的)还是对象,相应的内存地址中存放的都是一个数(无符号整数,整数,浮点数等)。传递参数的时候传递的就是相应内存地址中的数,所以说“ava函数是传值的”。当然,这个数对于基本类型和对象类型来说意义是不一样的,对于基本类型这个数就是其值本身,传递值的结果就是,改变新的变量的值不影响旧的变量的值;而对于对象来说这个数是它的地址,传递这个值就相当于传递了真实对象的引用,传递了引用或者说是地址的结果就是变化会全局可见,所以又可以说“java函数传递的参数是对象的引用”。

2.示例

代码示例:

public class TestRef {    public static void main(String[] args) {        ValueObject vo1 = new ValueObject("A", 1);        System.out.println("after vo1: " + vo1.getName()); // =A        changeValue1(vo1);        System.out.println("after changeValue1: " + vo1.getName()); // =A1, changed                                                                            changeValue2(vo1);        System.out.println("after changeValue2: " + vo1.getName()); // =A1,changeValue2内部的赋值不会影响这里。                                                                        }    /**     * 使用vo1自身的函数对其内部数据进行改变是有效的,函数外可反映出来,     * 因为这是对对象本身的操作 这种object称为可变的(mutable)     * @param vo1     */    private static void changeValue1(ValueObject vo1) {        vo1.setName("A1");    }    /**     * 在函数内给vo1重新赋值不会改变函数外的原始值,因为这种改变了引用的指向     * @param vo1     */    private static void changeValue2(ValueObject vo1) {        vo1 = new ValueObject("B", 2);        System.out.println("inside changeValue2: " + vo1.getName()); // =B,赋值操作引起的结果变化仅在changeValue2内部有效    }}class ValueObject {    private String name;    private int id;        public ValueObject() {    }    public ValueObject(String name, int id) {        this.name = name;        this.id = id;    }    public int getId() {        return id;    }    public void setId(int id) {        this.id = id;    }    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }}


java中对象的每个实例(就是对象)内存地址是唯一的,它一旦被创建,能够对这个地址进行操作的就是其本身,如果ValueObject类中没有public void setName之类的方法对这个类的实例中的数据进行修改的话,程序是没有任何别的方法可以修改ValueObject类的实例中的数据,这个就是java的封装特性。对于不提供修改内部数据的方法的类,我们称为不可变(immutable)的类。在函数中对传入的参数变量进行赋值操作,只能在函数范围内改变局部变量指向的引用地址,但是不会改变原始地址的内容。因此,在changeValue2(...)函数内部的vo1和函数外的vo1虽然名字相同,但是实际上是不同的实例变量,只不过指向了和函数外的vo1同样的地址,所以当我们用vo1=... 对其进行赋值的时候,只不过是把函数内的临时变量指向了新的地址(就是从原来vo1的地址变到了另外一个地址),并没有改变原始vo1内存地址中的内容。这就是在运行changeValue2(...)之后,vo1的值在main范围内仍然没有被修改的原因。而changeValue1里面是调用的ValueObject本身的function来更改其内容,因此是原始内存地址中的数据被更改了,所以是全局有效的.
总结:对于引用类型的传参也是传值的,传的是引用类型的值,其实就是对象的地址。

1. java参数传递值的。

2. java所有对像变量都是对像的引用。

 

 

0 0
原创粉丝点击