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所有对像变量都是对像的引用。
- Java参数传递,值传递
- Java参数传递
- Java参数传递
- java的参数传递
- java中的参数传递
- java中的参数传递
- JAVA的参数传递
- Java参数传递方式
- java参数传递
- Java-参数的传递
- java方法参数传递
- Java中的参数传递
- Java 参数传递机制
- java参数传递
- Java参数传递(推荐)
- java中的参数传递
- java参数传递
- Java的参数传递
- QWT绘制巴图
- ANSI C中常见的字符串函数
- 找工作必会必学
- Android关于notification的在不同API下的用法说明
- liblinear与libsvm学习资源
- java参数传递
- poj 2947 Widget Factory (高斯消元,解模线性方程)
- iOS用SegmentController来切换tableView的时候 reloadData会造成数据重复创建 解决办法
- 创建修改物料主数据的BAPI
- 注册账号的N多时间了,竟然没写过博客!
- 在windows上通过Spark访问HBase
- 关于函数strtok和strtok_r的使用要点和实现原理
- C - 剪花布条
- Android代码跳转到市场时,手机上没有市场软件的异常