java方法调用机制_Java方法调用机制 - osc_bkdv2it5的个人空间 - OSCHINA - 中文开源技术交流社区...

最近在编程时,修改方法传入对象的对象引用,并没有将修改反映到调用方法中。奇怪为什么结果没有变化,原因是遗忘了Java对象引用和内存分配机制。本文介绍3个点:

① 该问题举例说明

② 简要阐述Java内存区域

③ 介绍JVM中方法调用的机制

1. Java方法调用传参实例解析

Java中参数传递是值传递,即调用方法时,所有参数的传递都是值传递。基本类型直接将值拷贝给方法参数,引用类型将引用地址拷贝给方法参数。先看两个String类型和对象引用的实例。

(1)字符串对象引用

public static voidmain(String[] args) {

String a = "123";

app(a);

System.out.println(a);

}

private static voidapp(String a) {

//String不可修改,只会重新创建,故main中a不变

a += "456"; }

输出:123

分析:结果并没有因为调用了app方法,而输出123456。如注释中描述,String(由字符数组实现)是不可修改的,所有的修改都会重新创建新的String对象,并且字符串拼接也会重新创建String对象(具体见下文)。也就是说app中的a字符串引用已不再指向main中a指向的内存块,即main方法中a指向的内存块中字符串的值没有发送变化。

下图展示了对象引用与内存块的关系,可以看出来main方法中的与app方法中的a没关系,只是刚调用赋值的时候指向同一个内存块。

9a73921a0aa88d48e24421b8352e24d1.png

(2)字符串拼接源码实现

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

c5e01d1d3271811e4dc085e43d0dc090.png

读取的数据经第8行构建为String对象;

字符串拼接拆分为11-19,第11行构建StringBuilder对象,第16行调append方法,将字段串拼接到后边(底层通过将字符串中的字符放入原字符串字符数组中)

第19行,调StringBuilder.toString方法返回拼接好的字符串。

toString()的源码如下:

@Override

publicString 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, intto) {

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)); returncopy; }

(3)普通对象引用

public static voidmain(String[] args) {

ListNode node = new ListNode(4);

System.out.println(node);

chg(node);

System.out.println(node.val);

}

private static voidchg(ListNode node) {

node.val+=1;

System.out.println(node);

node= new ListNode(2); System.out.println(node); } static classListNode{ private intval; privateListNode next; ListNode(intval){ 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指向不同的对象。

40021dad8fc316a8cf90d32fca250da6.png

2. Java内存区域

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

JDK7中JVM运行时数据区的划分如下图:(JDK7时,已将字符串常量池从方法区移到堆中)

50c73649f498ad35cbfde8a9be54a6da.png

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

87f9bad86823619efe727db52d3693ca.png

3. Java方法调用机制(字节码执行引擎)

栈帧是支持虚拟机方法调用和方法执行的数据结构,每个方法调用都对应一个栈帧。栈帧中包含局部变量表、操作栈、动态连接和方法返回地址等信息,结构如下图所示(图摘自Java —— 运行时栈帧结构):

40a290276100aeb31466efadd1526221.png

具体内容的介绍参考书《深入理解Java虚拟机》。简单总结如下:

1. 基本概念

栈帧中局部变量表的大小、操作数栈的大小在编译期确定;

局部变量表用于存储方法参数和方法内部定义的局部变量,以槽(slot)为最小单位;

操作数栈用于存储指令计算对应的数据元素;

动态连接是指向运行时常量池中该栈帧所属方法放入引用,在运行期间可以转化为直接引用;

返回地址保存栈帧退出时,返回到方法被调用的位置,有正常退出(由PC计数器确定)和异常退出(由异常处理器表确定);

2. 方法调用

(1)解析

方法调用在Class文件中存储的都是符号引用,而不是方法在实际运行时内存布局中的入口地址(或直接引用)

在类加载的解析阶段,会将其中一部分符号引用转为直接引用,如编译期可知、运行期不可变的方法,包括静态方法和私有方法两大类,他们不可能被继承或重写;

5条字节码指令:invokestatic(调静态方法)、invokespecial(调构造方法、私有方法和父类方法)、invokevirtual(调虚方法)、invokeinterface(调接口方法,运行时确定实现该接口的对象)、invokedynamic(用于动态类型语言,暂不深究)。

前2种对应的为非虚方法,解析阶段可以将符号引用转为直接引用,不会延迟到运行期;invokevirtual和invokeinterface作用于虚方法(除final方法外)

(2)分派

Java多态性主要通过重载和重写实现,他们在JAVA虚拟机中是通过分派完成,包含静态分派(对应重载)和动态分派(对应重写)

定义一个变量,其有静态类型和实际类型,静态类型变量本身不会被改变,在编译期可知;编译时程序不知道一个对象实际类型是什么。

重载时,是通过参数的静态类型作为判断依据的;重写时,是在运行时根据实际类型作为判断依据,根据操作数栈顶所指的实际类型,去类型和父类型的常量池中查找对应方法。

3. 方法执行

Java虚拟机采用基于栈的字节码解释执行,过程涉及字节码指令、程序计数器、局部变量表和操作栈等,具体例子可参考书《深入理解Java虚拟机》。

4. 总结

方法调用时,方法参数是通过值传递的,并且方法参数会存储在栈帧中的局部变量表中,当修改该参数变量的指针时,与原来变量所指的内存块不同

Java虚拟机在JDK8时,将原来的永久代(方法区)改为元空间,放入本地内存。其中一个好处是防止永久代空间溢出问题

Java方法调用和执行是基于栈的字节码指令解释执行引擎,调用过程中涉及什么时机将符号引用转为直接引用,非虚方法调用发生在解析阶段,重载发生在编译期,重写发生在运行时。

5.参考

《深入理解Java虚拟机》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/370448.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CSS染色图标(图片)

之前一直以为用background引入的图标无法染色&#xff08;非字体图标&#xff09;&#xff0c;现在才知道有黑科技可以用&#xff0c;就是利用drop-shadow。 代码示例 <!DOCTYPE html> <html> <head lang"en"><meta charset"UTF-8"&…

eclipse安装java web插件

1 查看eclipse版本 找到eclipse的安装目录&#xff0c;找到readme文件&#xff0c;打开其中的html文件&#xff0c;我的是4.6版本的,代号是oxygen 2 安装 打开eclipse,点击help-Install new software-单击add&#xff0c;在弹出窗口中输入网址&#xff1a; http://download.ecl…

实现输入框小数多 自动进位展示,编辑时实际值不变

今天遇到个业务需求&#xff0c;要求输入框&#xff0c;输入数字的小数位数可以很多位&#xff0c;但移开后显示&#xff0c;只显示小数点后两位 &#xff08;四舍五入&#xff09;&#xff0c;当要编辑的时候&#xff0c;展现其原来的输入数据。 闲话不多说&#xff0c;当时也…

使用Jasper Reports以Java创建报告

上周&#xff0c;我试图使用Jasper创建报告。 在这篇文章中&#xff0c;我将记录一些资源和链接&#xff0c;以便对任何寻求类似信息的人都有用。 我将介绍Jasper报告&#xff0c;示例和Dynamic Jasper的生命周期。 Jasper Reports是世界上最受欢迎的开源报告引擎。 它完全用…

CentOS7 安装NodeJS

一、切换目录到/usr/local/src 命令行&#xff1a;cd /usr/local/src 二、下载node.js&#xff08;我这里下载的是二进制的源码&#xff09; 命令行&#xff1a; wget https://nodejs.org/dist/v8.9.1/node-v8.9.1-linux-x64.tar.xz 图片&#xff1a; 三、解压压缩包 命令行&am…

CSS3基础2(变形与动画)

<!DOCTYPE html5><html lang"en"><head> <meta charset"UTF-8"> <title>CSS3基础知识&#xff08;动画&#xff09;</title> <style> /*div{*/ /*width: 150px;*/ /*hei…

ASP.NET Core2.0 环境下MVC模式的支付宝PC网站支付接口-沙箱环境开发测试

1.新建.NET Core web项目 2.Controllers-Models-Views 分三个大部分 3.下载安装最新sdk 官方的SDK以及Demo都还是.NET Framework的&#xff0c;根据官方文档说明新建网站后还是需要引用官方SDK的源码&#xff0c; 在这里直接使用网上一位朋友的用.NET Standard 2.0 进行实现了支…

如何在redhat8里使用gcc命令_如何使用who命令检查用户登录信息

请关注本头条号&#xff0c;每天坚持更新原创干货技术文章。如需学习视频&#xff0c;请在微信搜索公众号“智传网优”直接开始自助视频学习1. 前言本教程主要介绍如何使用who命令检查用户登录信息。如何使用who命令检查用户登录信息Linux中的who命令列出了系统上的所有登录用户…

研究僵局–第4部分:修复代码

在这个简短的博客系列的最后BadTransferOperation中&#xff0c;我一直在讨论分析死锁&#xff0c;我将修复BadTransferOperation代码。 如果您看过本系列的其他博客 &#xff0c;那么您将知道&#xff0c;为了达到这一点&#xff0c;我创建了死锁的演示代码&#xff0c;展示了…

chrome插件2

转自&#xff1a;http://www.codeceo.com/article/15-chrome-extension.html 1. Web Developer 支持Chrome的Web Developer扩展&#xff0c;允许你通过添加一个小工具栏来使用不同的工具。 官方网站&#xff1a;https://chrome.google.com/webstore/detail/web-developer/bfbam…

java月历组件_vue之手把手教你写日历组件

---恢复内容开始---1.日历组件1.分析功能&#xff1a;日历基本功能&#xff0c;点击事件改变日期&#xff0c;样式的改变1.结构分析&#xff1a;html1.分为上下两个部分2.上面分为左按钮&#xff0c;中间内容展示&#xff0c;右按钮下面分为周几展示和日期展示3.基本结构页面ht…

maven project module 依赖项目创建 ---转

一、创建Maven Project 1.右击 --> New --> Other&#xff0c;--> Maven --> Maven Project --> Next 2.如下图&#xff0c;选中Create a simple project --> Next 3.输入Group Id, Artifact Id, Version, Packaging选择pom&#xff0c;因为创建的Maven Pr…

linux常见命令搜集

查找根目录下txt和pdf文件 find / \( -name "*.txt" -o -name "*.pdf" \) -print 正则查找根目录下所有的txt和pdf文件 find / -regex ".*\(\.txt|\.pdf\)$"查找所有非txt文本 find . ! -name "*.txt" -print制定搜索深度 find ~ -max…

需加装饰——装饰模式

装饰模式指的是在不必改变原类文件和使用继承的情况下&#xff0c;动态地扩展一个对象的功能。它是通过创建一个包装对象&#xff0c;也就是装饰来包裹真实的对象。 类图分析 我们先假设一个业务场景&#xff0c;有三种房子需要装修&#xff0c;分别是公寓&#xff0c;木屋和别…

Vue2.0 --- vue-cli脚手架中全局引入JQ

第一步&#xff1a;安装jQuery npm/cmpn方式安装(默认安装1.7.X版本的JQ) npm/cnpm install jQuery 如果想安装更高版本的JQ那么可以选择在package.json文件下面这个位置添加代码断&#xff08;当前图片安装的是2.2.3版本&#xff0c;如果想安装更高或者其他可以更改版本号&…

Unity——用UnityEditor拷贝FBX中的AnimationClip

最近有个新需求&#xff0c;要用代码添加动画的事件&#xff0c;但是Unity不能直接修改FBX中的AnimationClip 在Animation窗口中可以看到&#xff0c;AnimationClip是Read-Only状态&#xff0c;用代码修改这个AnimationClip也是不会生效的&#xff0c;包括用代码添加事件 解决方…

mvc如何嵌套第三方页面_长文观点丨为什么我不再使用MVC框架?

原创&#xff1a; 张卫滨 译 Jean-Jacques Dubray是一名资深工程师&#xff0c;他最近引入了一个新的模式&#xff1a;状态-行为-模(State-Action-Model&#xff0c;SAM)。SAM是一个函数式反应型的编程模式&#xff0c;它致力于简化数据Model和View之间的交互。它究竟有何优点值…

JSON和XML的区别

转载于:https://www.cnblogs.com/mr-wuxiansheng/p/6974239.html

屏幕适配

rem是什么&#xff1f; rem&#xff08;font size of the root element&#xff09;是指相对于根元素的字体大小的单位。简单的说它就是一个相对单位。看到rem大家一定会想起em单位&#xff0c;em&#xff08;font size of the element&#xff09;是指相对于父元素的字体大小…

【存储过程】MySQL存储过程/存储过程与自定义函数的区别

---------------------------存储过程-------------------- 语法: 创建存储过程: CREATE [definer {user|current_user}] PROCEDURE sp_name ([ proc_parameter [,proc_parameter ...]]) [ characteristics..] routime_body 其中: proc_parameter : [IN|OUT|INOUT] parameter_…