String是值传递还是引用传递
String是值传递还是引用传递
今天上班时,同事发现了一个比较有意思的问题。他把一个String类型的参数传入方法,并在方法内改变了引用的值。
然后他在方法外使用这个值,发现这个String还是之前的值,并没有改变。
这里要向大家介绍一下,大家都知道java在传参时分为值 传递 和 引用传递 。参数为基本类型时是值传递,
参数为封装类型时是引用传递。例如:
基本类型参数
public class Test { public static void main(String[] args) { int num = 0 ; changeNum(num); System.out.println("num="+num); } private static void changeNum(int num) { num = 1; } }
打印的结果是num=0
。
封装类型参数
public class Test { public static void main(String[] args) { Product p = new Product(); p.setProName("before"); p.setNum(0); changeProduct(p); System.out.println("p.proName="+p.getProName()); System.out.println("p.num="+p.getNum()); } private static void changeProduct(Product p) { p.setProName("after"); p.setNum(1); } } class Product { private int num; private String proName; public int getNum() { return num; } public void setNum(int num) { this.num = num; } public String getProName() { return proName; } public void setProName(String proName) { this.proName = proName; } }
运行的结果是:p.proName=after
和p.num=1
。
上面的两个例子是明显的值传递和引用传递。但是如果参数是String类型呢?我们看一下具体的例子:
public class Test { public static void main(String[] args) { String str = "ab"; changeString(str); System.out.println("str="+str); } private static void changeString(String str) { str = "cd"; } }
大家猜一下运行结果是什么呢?按照前面的例子,String应该是一个封装类型,它应该是引用传递,是可以改变值得,
运行的结果应该是"cd"。我们实际运行一下看看,
str=ab
,这如何解释呢?难道String是基本类型?也说不通呀。
这就要从java底层的机制讲起了,java的内存模型分为 堆 和 栈 。
1.基本类型的变量放在栈里; 2.封装类型中,对象放在堆里,对象的引用放在栈里。
java在方法传递参数时,是将变量复制一份,然后传入方法体去执行。 这句话是很难理解的,也是解释这个
问题的精髓。我们先按照这句话解释一下基本类型的传递
- 虚拟机分配给num一个内存地址,并且存了一个值0.
- 虚拟机复制了一个num,我们叫他num',num'和num的内存地址不同,但存的值都是0。
- 虚拟机讲num'传入方法,方法将num'的值改为1.
- 方法结束,方法外打印num的值,由于num内存中的值没有改变,还是0,所以打印是0.
我们再解释封装类型的传递:
- 虚拟机在堆中开辟了一个Product的内存空间,内存中包含proName和num。
- 虚拟机在栈中分配给p一个内存地址,这个地址中存的是1中的Product的内存地址。
- 虚拟机复制了一个p,我们叫他p',p和p'的内存地址不同,但它们存的值是相同的,都是1中Product的内存地址。
- 将p'传入方法,方法改变了1中的proName和num。
- 方法结束,方法外打印p中变量的值,由于p和p'中存的都是1中Product的地址,但是1中Product里的值发生了改变,
所以,方法外打印p的值,是方法执行以后的。我们看到的效果是封装类型的值是改变的。
最后我们再来解释String在传递过程中的步骤:
- 虚拟机在堆中开辟一块内存,并存值"ab"。
- 虚拟机在栈中分配给str一个内存,内存中存的是1中的地址。
- 虚拟机复制一份str,我们叫str',str和str'内存不同,但存的值都是1的地址。
- 将str'传入方法体
- 方法体在堆中开辟一块内存,并存值"cd"
- 方法体将str'的值改变,存入5的内存地址
- 方法结束,方法外打印str,由于str存的是1的地址,所有打印结果是"ab"
这样我们理解了java在方法传参的整个过程。其实还是上面那句比较重要的话 java在方法传递参数时,是将变量复制一份,然后传入方法体去执行。
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
JAVA9模块化详解(二)——模块的使用
JAVA9模块化详解(二)——模块的使用 二、模块的使用 各自的模块可以在模块工件中定义,要么就是在编译期或者运行期嵌入的环境中。为了提供可靠的配置和强健的封装性,在分块的模块系统中利用他们,必须确定它们的位置,然后决定他们如何关联彼此。 2.1 模块的路径 为了在确定在工件中定义的模块的位置,模块系统搜索模块的路径,它在主系统中定义。模块路径是一个序列,它的每一个元素要么是一个模块工件,要么是一个包含模块工件的目录。模块路径中的元素被第一个工件有序的搜索,这个工件定义了一个合适的模块。模块的路径在物理上不同于类路径,而且更强大。类路径天生的脆弱性是它定位了路径下所有工件中的个体类型,在工件中间没有任何的区分。这使得它在工件丢失时不可能提前告知,它也允许不同的工件在相同的包中定义类型,即使那些工件代表中相同程序组件的不同版本,或者完全不同的组件。 相比之下,模块路径定位整个的模块,而不是个体类型。如果模块系统不能从模块路径中处理工件的特殊依赖,或者如果它在相同的目录下遇到了两个模块名字相同的工件,这是编译器或者虚拟机将报告一个错误并退出。 嵌入到编译器或运行期环境的模块,连同模块路径下...
- 下一篇
Mybatis Generator 使用com.mysql.cj.jdbc.Driver遇到的问题
Mybatis Generator 使用com.mysql.cj.jdbc.Driver遇到的问题 今天闲来无事,准备搭一套SSM的环境,当然所有的jar包都用最新的。 Mybatis使用3.4.6,Mysql使用最新的8.0,mysql-connector-java用的8.0.11。 安装好数据库后,新建了一个test库,并创建了一张user表。配置好Mybatis的配置文件,如下: <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE generatorConfiguration PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN" "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd" > <generatorConfiguration> <!-- 指定数据连接驱动jar地址 --> <classPathEntry location="D:...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Linux系统CentOS6、CentOS7手动修改IP地址
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- Windows10,CentOS7,CentOS8安装MongoDB4.0.16
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- CentOS7安装Docker,走上虚拟化容器引擎之路
- CentOS6,CentOS7官方镜像安装Oracle11G
- CentOS8编译安装MySQL8.0.19