网站首页 > java教程 正文
值传递
int n = 1;
int n1 = n;
n1 = 20;
System.out.println("n = " + n + " n1=" + n1);
上面n和n1的输出结果分别为1,20。这就说明n1值的改变,不影响n的值
基本数据类型之间赋值,属于值传递(值拷贝)
下面从jvm层面分析下原理。使用 javap -v xxx.class 命令,获取字节码文件的汇编指令
public static void main(java.lang.String[]) throws java.io.IOException;
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=3, locals=3, args_size=1
0: iconst_1 //将常量1压入栈中
1: istore_1 //常量1出栈,保存到局部变量表,槽点为1的位置
2: iload_1 //常量1入栈
3: istore_2 // 常量1出栈,保存到局部变量表, 槽点为2的位置
4: bipush 20 //将常量20压入栈中
6: istore_2 //常量20出栈,保存到局部变量表,槽点为2的位置
7: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
10: new #3 // class java/lang/StringBuilder
13: dup
14: invokespecial #4 // Method java/lang/StringBuilder."<init>":()V
17: ldc #5 // String n =
19: invokevirtual #6 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
22: iload_1
23: invokevirtual #7 // Method java/lang/StringBuilder.append:(I)Ljava/lang/StringBuilder;
26: ldc #8 // String n1=
28: invokevirtual #6 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
31: iload_2
32: invokevirtual #7 // Method java/lang/StringBuilder.append:(I)Ljava/lang/StringBuilder;
35: invokevirtual #9 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
38: invokevirtual #10 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
41: return
LineNumberTable: //行号表,代码的行号与指令索引的映射关系
line 58: 0
line 59: 2
line 60: 4
line 61: 7
line 79: 41
LocalVariableTable: //局部变量表
Start Length Slot Name Signature
0 42 0 args [Ljava/lang/String;
2 40 1 n I
4 38 2 n1 I
从字节码指令中,不难看出,变量n,n1分别存储在局部变量表的槽点1,2位置上。
- 执行 int n = 1 ,在LocalVariableTable中的槽点1的位置存放变量n,值为1
- 执行 int n1 = n ,在LocalVariableTable中的槽点2的位置存放变量n1,值为1
- 执行 n1 = 20 ,将20赋值给槽点2的变量n1,槽点1中的变量n的值不变。
以上得出,基本数据类型赋值,是值传递,后面再对任一变量修改,其实改的是不同地方,所以互不影响。
引用传递
int[] arr = {1,23,3};
int[] arr1 = arr;
arr1[0] = 120;
问题:
上面代码执行完了,数组arr的第一个元素的值是否也变成了120?
下面咱们从字节码指令进行分析
public static void main(java.lang.String[]) throws java.io.IOException;
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=4, locals=3, args_size=1
0: iconst_3 // 常量3入栈
1: newarray int // new一个长度为3的数组
3: dup // 复制数组对象引用地址入栈
4: iconst_0 // 数组角标常量0入栈
5: iconst_1 // 常量1入栈
6: iastore // 出栈, 将栈顶int型数值1存入数组的索引0的位置
7: dup // 复制数组对象引用地址入栈
8: iconst_1 // 数组角标常量1入栈
9: bipush 23 // // 常量23入栈
11: iastore // 出栈, 将栈顶int型数值23存入数组的索引1的位置
12: dup // 复制数组对象引用地址入栈
13: iconst_2 // 数组角标常量2入栈
14: iconst_3 // 常量3入栈
15: iastore // 出栈, 将栈顶int型数值3存入数组的索引2的位置
16: astore_1 // 数组对象引用地址出栈,保存到局部变量表,槽点1的位置,变量名为arr
17: aload_1 // 槽点1中的变量arr引用地址入栈
18: astore_2 // 栈顶变量arr引用地址出栈,保存到局部变量表,槽点2的位置,变量名称为arr1
19: aload_2 // 槽点2中的变量arr1引用地址入栈
20: iconst_0· // 角标常量0入栈
21: bipush 120 //常量120入栈
23: iastore // 出栈, 将栈顶int型数值120存入数组的索引0的位置
24: return
LineNumberTable:
line 63: 0
line 64: 17
line 65: 19
line 77: 24
LocalVariableTable:
Start Length Slot Name Signature
0 25 0 args [Ljava/lang/String;
17 8 1 arr [I
19 6 2 arr1 [I
Exceptions:
throws java.io.IOException
MethodParameters:
Name Flags
args
从字节码指令中可以看出,数组arr,arr1都指向了同一个数组对象地址,当对数组某个角标的值进行改变,引用这个数组的所有变量的值都发生了改变。
猜你喜欢
- 2024-11-11 挨个举例子告诉你Java中的参数传递,我就不信你还不明白了
- 2024-11-11 Java值传递机制(java传值方式)
- 2024-11-11 Java面试-为什么 Java 只有值传递?
- 2024-11-11 Stack Overflow上188W+程序员都关注的问题
- 2024-11-11 为什么说Java不存在引用传递?(javax不存在)
- 2024-11-11 一份2021年的java面试题(java常见面试题)
- 2024-11-11 互联网大厂面试系列-为什么说在Java方法参数调用只有值传递呢?
- 2024-11-11 一文搞懂参数传递原理(参数传递的三种方式)
- 2024-11-11 Java程序员想年后跳槽,对JVM没有深入的理解,我劝你还是别跳了
- 2024-11-11 JAVA校招题基础知识点复习第二天(引用传递的本质是什么?)
你 发表评论:
欢迎- 最近发表
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)