Java方法调用机制
最近在编程时,修改方法传入对象的对象引用,并没有将修改反映到调用方法中。奇怪为什么结果没有变化,原因是遗忘了Java对象引用和内存分配机制。本文介绍3个点:
① 该问题举例说明
② 简要阐述Java内存区域
③ 介绍JVM中方法调用的机制
1. Java方法调用传参实例解析
Java中参数传递是值传递,即调用方法时,所有参数的传递都是值传递。基本类型直接将值拷贝给方法参数,引用类型将引用地址拷贝给方法参数。先看两个String类型和对象引用的实例。
(1)字符串对象引用
public static void main(String[] args) {
String a = "123";
app(a);
System.out.println(a);
}
private static void app(String a) {
//String不可修改,只会重新创建,故main中a不变
a += "456";
}
输出:123
分析:结果并没有因为调用了app方法,而输出123456。如注释中描述,String(由字符数组实现)是不可修改的,所有的修改都会重新创建新的String对象,并且字符串拼接也会重新创建String对象(具体见下文)。也就是说app中的a字符串引用已不再指向main中a指向的内存块,即main方法中a指向的内存块中字符串的值没有发送变化。
下图展示了对象引用与内存块的关系,可以看出来main方法中的与app方法中的a没关系,只是刚调用赋值的时候指向同一个内存块。

(2)字符串拼接源码实现
通过下图中字节码命令可以看到,字符串拼接是通过StringBuilder实现的。

- 读取的数据经第8行构建为String对象;
- 字符串拼接拆分为11-19,第11行构建StringBuilder对象,第16行调append方法,将字段串拼接到后边(底层通过将字符串中的字符放入原字符串字符数组中)
- 第19行,调StringBuilder.toString方法返回拼接好的字符串。
toString()的源码如下:
@Override
public String toString() {
// Create a copy, don't share the array
return new String(value, 0, count);
}
重新构建String对象,并且注释中说明创建拷贝,但不共享字符数组。String构造函数底层会调Arrays.copyOfRange(char[] original, int from, int to)方法,将original字符数组拷贝到一个新的字符数组中,源码如下:
public static char[] copyOfRange(char[] original, int from, int to) {
int newLength = to - from;
if (newLength < 0)
throw new IllegalArgumentException(from + " > " + to);
char[] copy = new char[newLength];
System.arraycopy(original, from, copy, 0,
Math.min(original.length - from, newLength));
return copy;
}
(3)普通对象引用
public static void main(String[] args) {
ListNode node = new ListNode(4);
System.out.println(node);
chg(node);
System.out.println(node.val);
}
private static void chg(ListNode node) {
node.val+=1;
System.out.println(node);
node= new ListNode(2);
System.out.println(node);
}
static class ListNode{
private int val;
private ListNode next;
ListNode(int val){
this.val = val;
}
}
输出:
test.InsertSortTest$ListNode@2a139a55
test.InsertSortTest$ListNode@2a139a55
test.InsertSortTest$ListNode@15db9742
5
分析:① test.InsertSortTest$ListNode@2a139a55地址对应的对象,在main方法调chg方法传递参数的时候,将地址拷贝给chg方法的参数node,在chg方法中修改对象的值,此时两个方法中的node仍指向统一内存块,故main方法中输出为5。
② node= new ListNode(2) 语句将chg方法中的node重新指向另一个对象地址test.InsertSortTest$ListNode@15db9742,此时main方法和chg方法中的node指向不同的对象。

2. Java内存区域
具体Java虚拟机运行时数据区的划分,网上有很多相关资料,还可以看《深入理解Java虚拟机》,在此就不赘述了。但需要说明一点JDK7和JDK8稍有不同,就是JDK8中将原有的方法区(Method Area)或永久代改为元空间(MetaSpace),即将存储类信息、静态变量等元数据信息从方法区(也是堆内存)移动到本地内存(native memory)中。将不会出现java.lang.OutOfMemoryError: PermGen异常,如果该区域设置了大小,可能会出现java.lang.OutOfMemoryError: Metadata space异常,如果不设置大小,默认是自增的。
JDK7中JVM运行时数据区的划分如下图:(JDK7时,已将字符串常量池从方法区移到堆中)

JDK8中JVM运行时数据区的划分:

这两张图分别参考自选择JDK1.8的理由之JVM内存变化和深入理解系列之JDK8下JVM虚拟机(1)——JVM内存组成,其中对变化也阐述的比较清楚。
3. Java方法调用机制(字节码执行引擎)
栈帧是支持虚拟机方法调用和方法执行的数据结构,每个方法调用都对应一个栈帧。栈帧中包含局部变量表、操作栈、动态连接和方法返回地址等信息,结构如下图所示(图摘自Java —— 运行时栈帧结构):

具体内容的介绍参考书《深入理解Java虚拟机》。简单总结如下:
1. 基本概念
- 栈帧中局部变量表的大小、操作数栈的大小在编译期确定;
- 局部变量表用于存储方法参数和方法内部定义的局部变量,以槽(slot)为最小单位;
- 操作数栈用于存储指令计算对应的数据元素;
- 动态连接是指向运行时常量池中该栈帧所属方法放入引用,在运行期间可以转化为直接引用;
- 返回地址保存栈帧退出时,返回到方法被调用的位置,有正常退出(由PC计数器确定)和异常退出(由异常处理器表确定);
2. 方法调用
(1)解析
- 方法调用在Class文件中存储的都是符号引用,而不是方法在实际运行时内存布局中的入口地址(或直接引用)
- 在类加载的解析阶段,会将其中一部分符号引用转为直接引用,如编译期可知、运行期不可变的方法,包括静态方法和私有方法两大类,他们不可能被继承或重写;
- 5条字节码指令:invokestatic(调静态方法)、invokespecial(调构造方法<init>、私有方法和父类方法)、invokevirtual(调虚方法)、invokeinterface(调接口方法,运行时确定实现该接口的对象)、invokedynamic(用于动态类型语言,暂不深究)。
- 前2种对应的为非虚方法,解析阶段可以将符号引用转为直接引用,不会延迟到运行期;invokevirtual和invokeinterface作用于虚方法(除final方法外)
(2)分派
- Java多态性主要通过重载和重写实现,他们在JAVA虚拟机中是通过分派完成,包含静态分派(对应重载)和动态分派(对应重写)
- 定义一个变量,其有静态类型和实际类型,静态类型变量本身不会被改变,在编译期可知;编译时程序不知道一个对象实际类型是什么。
- 重载时,是通过参数的静态类型作为判断依据的;重写时,是在运行时根据实际类型作为判断依据,根据操作数栈顶所指的实际类型,去类型和父类型的常量池中查找对应方法。
3. 方法执行
Java虚拟机采用基于栈的字节码解释执行,过程涉及字节码指令、程序计数器、局部变量表和操作栈等,具体例子可参考书《深入理解Java虚拟机》。
4. 总结
- 方法调用时,方法参数是通过值传递的,并且方法参数会存储在栈帧中的局部变量表中,当修改该参数变量的指针时,与原来变量所指的内存块不同
- Java虚拟机在JDK8时,将原来的永久代(方法区)改为元空间,放入本地内存。其中一个好处是防止永久代空间溢出问题
- Java方法调用和执行是基于栈的字节码指令解释执行引擎,调用过程中涉及什么时机将符号引用转为直接引用,非虚方法调用发生在解析阶段,重载发生在编译期,重写发生在运行时。
5.参考
《深入理解Java虚拟机》
Java方法调用机制的更多相关文章
- java 方法调用绑定
将一个方法调用同一个方法主体关联起来被称作绑定.若在程序执行前进行绑定(由编译器和连接器实现),叫做前期绑定.读者可能从来没有听说过这个术语,因为它在面向过程语言中不需要选择就默认的绑定方式.例如C语 ...
- java方法调用之动态调用多态(重写override)的实现原理——方法表(三)
上两篇篇博文讨论了java的重载(overload)与重写(override).静态分派与动态分派.这篇博文讨论下动态分派的实现方法,即多态override的实现原理. java方法调用之重载.重写的 ...
- JAVA方法调用中的解析与分派
JAVA方法调用中的解析与分派 本文算是<深入理解JVM>的读书笔记,参考书中的相关代码示例,从字节码指令角度看看解析与分派的区别. 方法调用,其实就是要回答一个问题:JVM在执行一个方法 ...
- 难道同事:Java 方法调用到底是传值还是传引用
Java 方法调用中的参数是值传递还是引用传递呢?相信每个做开发的同学都碰到过传这个问题,不光是做 Java 的同学,用 C#.Python 开发的同学同样肯定遇到过这个问题,而且很有可能不止一次. ...
- 难住了同事:Java 方法调用到底是传值还是传引用
Java 方法调用中的参数是值传递还是引用传递呢?相信每个做开发的同学都碰到过传这个问题,不光是做 Java 的同学,用 C#.Python 开发的同学同样肯定遇到过这个问题,而且很有可能不止一次. ...
- Java方法调用数组,是否改变原数组元素的总结
Java方法调用数组,是否改变原数组元素的总结 //个人理解, 欢迎吐槽 注意String是引用型变量, 我的理解也就是指向型, 指向一个数据或变量, 画图理解最容易, string 指向的 数据的值 ...
- 第47篇-解释执行的Java方法调用native方法小实例
举个小实例,如下: public class TestJNI { static { // 程序在加载时,自动加载libdiaoyong.so库 System.loadLibrary("dia ...
- 【Java基础】方法调用机制——MethodHandle
MethodHandle是Java7引入的一种机制,主要是为了JVM支持动态语言. 一个MethodHandle调用示例 共有方法调用 首先,演示一下最基本的MethodHandle使用. 第一步:创 ...
- java 方法调用绑定--《java编程思想》学习笔记
将一个方法调用同一个方法主体关联起来,就是绑定. 绑定分两种 :前期绑定 和 后期绑定 . 绑定------------- | -----前期绑定-------编译期绑定 { static , fin ...
随机推荐
- 查找担保圈-step6-对被包含过的组进行清理,只保留未被包含过的组
USE [test] GO /****** Object: StoredProcedure [dbo].[p04_get_groupno_cleared] Script Date: 2019/7/8 ...
- Spring的四种事务特性,五种隔离级别,七种传播行为
Spring事务: 什么是事务: 事务逻辑上的一组对数据对操作,组成这些操作的各个逻辑单元,要么一起成功,要么一起失败. 事务特性(4种): 原子性(atomicity):强调事务的不可分割:一致性( ...
- linux常用国内的免费源及其各别的配置方法.阿里源,epel源,搜狐网易等等..
国内的一些开源的平台收集的源,确实给我们提供了很多便利,所以我就稍微收集整理了一些常用的源和网址,我也不确定能用到什么时候!欢迎评论区留言! 搜狐开源镜像站 http://mirrors.sohu.c ...
- python私有化xx、_xx、__xx、__xx__、xx_的区别
xx:共有变量. _xx:私有化的属性或方法,from xxx import * 时无法导入,子类的对象和子类可以访问. __xx:避免与子类中的属性命名冲突,无法在外部直接访问(名字重整所以访问不到 ...
- IDEA 中git的分支管理和使用说明
1. 为什么要建立分支 git默认的主分支名字为master,一般团队开发时,都不会在master主分支上修改代码,而是建立新分支,测试完毕后,在将分支的代码合并到master主分支上. 2.操作如下 ...
- 二叉查找树 平衡二叉查找树 红黑树 b树 b+树 链表 跳表 链表
https://www.cnblogs.com/mojxtang/p/10122587.html二叉树的新增遍历查找
- 关于学习电信nb-iot的小结
关于这几天对nb-iot的学习的总结和遇到的坑 初步学习nb-iot,了解到了nb-iot对于传感器数据传输功能的强大: 废话不多说,对于nb-iot我们选择的有人的模块,选择B5频段也就是电信的nb ...
- day06 Python class基础篇
一.目录 1.类与对象的概述 2.封装 3.继承 4.多态 5.类的成员 6.类与类之间的关系 7.私有 二. 内容讲解 一.类与对象的概述 类是对一系列具有相同属性的事物的抽象,相同于设计图纸,而对 ...
- java指定运行jar包中的其中一个main方法
java -cp jar包 类名 java -cp ******.jar com.******.EsEtl
- Oracle子句【group by、having】
[分组查询]关键字:group by 分组字段名,分组字段名... --注意1:分组后,在select语句中只允许出现分组字段和多行函数 --注意2:如果是多字段分组,先按第一字段分组,然后每个小组继 ...