网站首页 > 基础教程 正文
前言
今天做项目,发现了一个问题,当String作为参数传递的时候,在函数内部改变值对外部的变量值无影响,如下代码:
public static void main(String[] args) {
String str = "11111";
changeStr(str);
System.out.println(str);//输出11111
}
public static void changeStr(String str){
str = "22222";
}
一开始不解后来想想就明白了,遂去对java参数传递做一个总结,以加深自己的基础知识.
1.基本类型传递
对于:
- 整型: byte short int long
- 浮点型 float double
- 逻辑型 boolean
- 字符型 char 四类八种基本类型来说,传递的都是值,因为这些值是直接保存在栈内存中的,所以传递的时候直接拷贝过去了.
public static void main(String[] args) {
int num = 0;
change(num);
System.out.println(num);//输出0
}
public static void change(int numChange){
numChange =5;
}
结构如下图,也因此最外层的num的值并没有受到影响.
2.对象传递
2.1 例一
对象传递,本质上也都是值传递,只不过传递的值是该引用的拷贝.看下面实例和图解:
public static void main(String[] args) {
Person person = new Person("aaa", 111);
change(person);
System.out.println(person);//输出 bbb 111
}
public static void change(Person personChange){
personChange.setName("bbb");
}
结构图如下: 当执行change的时候,会把person变量的指向的地址拷贝一份给personChange,两者都指向同一个堆内存,即使后面做了set方法修改,但是对两者的执行毫无影响.
2.2例二
例二和之前的不同之处在change里面,对personChange进行了new操作.代码如下:
public static void main(String[] args) {
Person person = new Person("aaa", 111);
change(person);
System.out.println(person);//输出 aaa 111
}
public static void change(Person personChange){
personChange = new Person("bbb",222);
}
结构图如下: 当执行change的时候,会把person变量的指向的地址拷贝一份给personChange,两者都指向同一个堆内存,接下new操作会在堆中重新创建一个person对象,此时personChange则指向这个对象,而原person的指向没发生变化,故输出aaa 111
2.3 例三
例三是综合例一和例二,前面两个搞懂的话这个就很容易懂了.
public static void main(String[] args) {
Person person = new Person("aaa", 111);
change(person);
System.out.println(person);//输出 ccc 111
}
public static void change(Person personChange){
personChange.setName("ccc");
personChange = new Person("bbb",222);
}
结构图如下: 读者自己理解下,不懂的话再看看前面的,看看为什么输出CCC 111
2.4特殊的String
终于到最初的问题,为什么String是对象,但是却不符合上面对象传递测试出来的结果?
public static void main(String[] args) {
String str = "11111";
changeStr(str);
System.out.println(str);//输出11111
}
public static void changeStr(String str){
str = "22222";
}
原因: 因为String对象具有不可变性,所以针对操作str = "22222",在String池中不存在的时候,就是相当于str = new String(),这样变化下的话,那么就和例一 一模一样了,具体图就不画了,希望对你有帮助.
ps:如果想改变的话,可以使用Holder包装类包装String
3.总结
要理解上面的结果,就要认为Java中只有值传递:
- 对于基本类型,直接拷贝值传递过去
3.总结
要理解上面的结果,就要认为Java中只有值传递:
- 对于基本类型,直接拷贝值传递过去
- 对于对象,拷贝当前对象的引用地址,然后把该地址传递过去,所以也是值传递.
最后
大家看完有什么不懂的可以在下方留言讨论.
谢谢你的观看。
觉得文章对你有帮助的话记得关注我点个赞支持一下!
作者:衍方
链接:
https://juejin.im/post/6869278349177569288
- 上一篇: java 整型类型_Java基本类型-整型解读
- 下一篇: Java运行时数据区域
猜你喜欢
- 2025-03-13 python:python与java语法的异同之处
- 2025-03-13 Java 性能优化的 50 个细节(珍藏版)
- 2025-03-13 详解Java泛型之4——一个例子理解泛型带来的好处
- 2025-03-13 几道小小的题目一起和大家更进一步的了解 Java (1)
- 2025-03-13 C++语言的单元测试与代码覆盖率
- 2025-03-13 java互联网架构师,教你学java基本的程序设计结构:大数值+数组
- 2025-03-13 三十六、Java包装类
- 2025-03-13 Java泛型全方位剖析:从入门到精通的完整指南(上篇)
- 2025-03-13 自学Java2(保姆级教学)——常量与变量
- 2025-03-13 JAVA反射机制详解,一学就会
- 最近发表
-
- 掌握SpringBoot-2.3的容器探针:实战篇
- kubernetes基础知识之驱逐节点(k8s驱逐节点后恢复)
- Linux环境中制作网络哨兵Sentinel Docker镜像
- k8s之配置CNI网络(k8s 网络配置)
- docker实战之:镜像更新(docker 镜像升级替换)
- 离线在docker镜像方式部署ragflow0.17.2
- Linux日常小技巧Docker打包(docker打包lnmp)
- 使用dockerfile构建docker镜像(docker通过dockerfile构建镜像命令)
- 「云原生」Containerd ctr,crictl 和 nerdctl 命令介绍与实战操作
- Kylin安装Dify(kylin安装部署)
- 标签列表
-
- jsp (69)
- pythonlist (60)
- gitpush (78)
- gitreset (66)
- python字典 (67)
- dockercp (63)
- gitclone命令 (63)
- dockersave (62)
- pythonif (68)
- pythonifelse (59)
- deletesql (62)
- c++模板 (62)
- c#event (59)
- linuxgzip (68)
- 字符串连接 (73)
- nginx配置文件详解 (61)
- html标签 (69)
- c++初始化列表 (64)
- exec命令 (59)
- mysqlinnodbmyisam区别 (63)
- arraylistadd (66)
- console.table (62)
- mysqldatesub函数 (63)
- window10java环境变量设置 (66)
- c++虚函数和纯虚函数的区别 (66)