- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章浅谈从Java中的栈和堆,进而衍生到值传递由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
简述Java中的栈和堆,变量和对象的地址存放和绑定机制 。
初学java的小白,很多人都搞不清楚java中堆和栈的概念,我们都知道计算机只能识别二进制字节码文件,如果分不清楚对象和变量在内存的地址分配,也就是堆和栈的问题,很多问题比如绑定机制、静态方法、实例方法、局部变量的作用域就会搞不清楚.
首先记住结论:
基本数据类型、局部变量、String类型的直接赋值都是存放在栈内存中的,用完就消失.
new创建的实例化对象、String类型的构造方法new出来的对象及数组,是存放在堆内存中的,用完之后靠垃圾回收机制不定期自动消除.
地址是栈,就是静态绑定机制,执行完值不变化;地址是堆(对象在堆内储存,一般也会在栈里分配一个空间,去指向堆里的对象的地址)就是动态绑定机制,执行完值变化.
栈和堆 。
栈:基本类型变量,String类型的直接赋值变量,对象的实例变量都在函数的栈内存中分配。栈内存特点,数据一执行完毕,变量会立即释放,节约内存空间;并且必须初始化变量的值.
堆:堆内存用来存放new创建的对象、String类型的构造方法new出来的对象和数组。堆内存中所有的实体都有内存地址值,系统会自动初始化变量的值;当堆内存中的实体不再被指向时,JVM启动垃圾回收机制,自动清除.
举例1:
1
2
3
4
5
6
7
8
9
10
11
|
public
static
void
main(String[] args) {
int
sum =
0
;
String str =
"abc"
;
for
(
int
i =
1
; i<score; i++){
sum += i;
}
//可以打印sum
System.out.println(sum);
不可以打印i
System.out.println(i);
}
|
如下图所示:
以上程序执行步骤:
第1步——main()函数是程序入口,JVM先执行,在栈内存中开辟链两个空间,存放int类型变量sum,同时附值0;String类型变量 str,并赋值"abc"; 。
第2步——JVM执行for循环是,在栈内存中又开辟一个新的空间,存放int类型变量i,同时附值1。 此时main空间与for空间并存,同时运行,互不影响。 第3步——for()执行完毕,变量i立即释放,空间消失。但是main()函数空间仍存在,main中的变量sum和str仍然存在,不受影 响.
从上可以看出:基本数据类型、局部变量、String类型的直接赋值都是存放在栈内存中的,用完就消失。地址是栈,就是静态绑定机制,执行完值不变化.
举例2:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public
class
Test1 {
int
score;
public
static
void
main(String[] args) {
int
[] sum = {
0
,
1
,
2
};
String str =
new
String(
"abc"
);
Test1 test1 =
new
Test1();
test1.score =
98
;
test1.showInfo();
}
public
void
showInfo(){
System.out.println(
"我的成绩是"
+score);
}
}
|
上述代码的意思如下图所示:
从上可以看出:new创建的实例化对象、String类型的构造方法new出来的对象及数组,是存放在堆内存中的,用完之后靠垃圾回收机制不定期自动消除。地址是堆(对象在堆内储存,一般也会在栈里分配一个空间,去指向堆里的对象的地址)就是动态绑定机制,执行完值变化.
总结:
基本数据类型、局部变量、String类型的直接赋值都是存放在栈内存中的,用完就消失.
new创建的实例化对象、String类型的构造方法new出来的对象及数组,是存放在堆内存中的,用完之后靠垃圾回收机制不定期自动消除.
地址是栈,就是静态绑定机制,执行完值不变化;地址是堆(对象在堆内储存,一般也会在栈里分配一个空间,去指向堆里的对象的地址)就是动态绑定机制,执行完值变化.
值传递 。
到底是值传递还是引用传递 。
是值传递。Java 语言的方法调用只支持参数的值传递。当一个对象实例作为一个参数被传递到方法中时,参数的值就是对该对象的引用。对象的属性可以在被调用过程中被改变,但对对象引用的改变是不会影响到调用者的 。
为什么 Java 中只有值传递 。
首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语。按值调用(call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址。一个方法可以修改传递引用所对应的变量值,而不能修改传递值调用所对应的变量值。 它用来描述各种程序设计语言(不只是Java)中方法参数传递方式.
Java程序设计语言总是采用按值调用。也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参数变量的内容.
下面通过 3 个例子来给大家说明 。
example 1 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
public
static
void
main(String[] args) {
int
num1 =
10
;
int
num2 =
20
;
swap(num1, num2);
System.out.println(
"num1 = "
+ num1);
System.out.println(
"num2 = "
+ num2);
}
public
static
void
swap(
int
a,
int
b) {
int
temp = a;
a = b;
b = temp;
System.out.println(
"a = "
+ a);
System.out.println(
"b = "
+ b);
}
|
结果:
a = 20 b = 10 num1 = 10 num2 = 20 。
在swap方法中,a、b的值进行交换,并不会影响到 num1、num2。因为,a、b中的值,只是从 num1、num2 的复制过来的。也就是说,a、b相当于num1、num2 的副本,副本的内容无论怎么修改,都不会影响到原件本身.
通过上面例子,我们已经知道了一个方法不能修改一个基本数据类型的参数,而对象引用作为参数就不一样,请看 example2. 。
example 2 。
1
2
3
4
5
6
7
8
9
10
11
|
public
static
void
main(String[] args) {
int
[] arr = {
1
,
2
,
3
,
4
,
5
};
System.out.println(arr[
0
]);
change(arr);
System.out.println(arr[
0
]);
}
public
static
void
change(
int
[] array) {
// 将数组的第一个元素变为0
array[
0
] =
0
;
}
|
结果:
1 0 。
解析:
array 被初始化 arr 的拷贝也就是一个对象的引用,也就是说 array 和 arr 指向的时同一个数组对象。 因此,外部对引用对象的改变会反映到所对应的对象上.
通过 example2 我们已经看到,实现一个改变对象参数状态的方法并不是一件难事。理由很简单,方法得到的是对象引用的拷贝,对象引用及其他的拷贝同时引用同一个对象.
很多程序设计语言(特别是,C++和Pascal)提供了两种参数传递的方式:值调用和引用调用。有些程序员(甚至本书的作者)认为Java程序设计语言对对象采用的是引用调用,实际上,这种理解是不对的。由于这种误解具有一定的普遍性,所以下面给出一个反例来详细地阐述一下这个问题.
example 3 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
public
class
Test {
public
static
void
main(String[] args) {
// TODO Auto-generated method stub
Student s1 =
new
Student(
"小张"
);
Student s2 =
new
Student(
"小李"
);
Test.swap(s1, s2);
System.out.println(
"s1:"
+ s1.getName());
System.out.println(
"s2:"
+ s2.getName());
}
public
static
void
swap(Student x, Student y) {
Student temp = x;
x = y;
y = temp;
System.out.println(
"x:"
+ x.getName());
System.out.println(
"y:"
+ y.getName());
}
}
|
结果:
x:小李 y:小张 s1:小张 s2:小李 。
解析:
交换之前:
交换之后:
通过上面两张图可以很清晰的看出: 方法并没有改变存储在变量 s1 和 s2 中的对象引用。swap方法的参数x和y被初始化为两个对象引用的拷贝,这个方法交换的是这两个拷贝 。
总结 。
Java程序设计语言对对象采用的不是引用调用,实际上,对象引用是按值传递的.
下面再总结一下Java中方法参数的使用情况:
一个方法不能修改一个基本数据类型的参数(即数值型或布尔型》 。
一个方法可以改变一个对象参数的状态.
一个方法不能让对象参数引用一个新的对象.
值传递和引用传递有什么区别 。
值传递:指的是在方法调用时,传递的参数是按值的拷贝传递,传递的是值的拷贝,也就是说传递后就互不相关了.
引用传递:指的是在方法调用时,传递的参数是按引用进行传递,其实传递的引用的地址,也就是变量所对应的内存空间的地址。传递的是值的引用,也就是说传递前和传递后都指向同一个引用(也就是同一个内存空间).
以上这篇浅谈从Java中的栈和堆,进而衍生到值传递就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我.
原文链接:https://blog.csdn.net/BigData_Hobert/article/details/106161876 。
最后此篇关于浅谈从Java中的栈和堆,进而衍生到值传递的文章就讲到这里了,如果你想了解更多关于浅谈从Java中的栈和堆,进而衍生到值传递的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
服务架构进化论 原始分布式时代 一直以来,我可能和大多数的人认知一样,认为我们的服务架构的源头是单体架构,其实不然,早在单体系
序列化和反序列化相信大家都经常听到,也都会用, 然而有些人可能不知道:.net为什么要有这个东西以及.net frameword如何为我们实现这样的机制, 在这里我也是简单谈谈我对序列化和反序列化的
内容,是网站的核心所在。要打造一个受用户和搜索引擎关注的网站,就必须从网站本身的内容抓起。在时下这个网络信息高速发展的时代,许多低质量的信息也在不断地充斥着整个网络,而搜索引擎对一些高质量的内容
从第一台计算机问世到现在计算机硬件技术已经有了很大的发展。不管是现在个人使用的PC还是公司使用的服务器。双核,四核,八核的CPU已经非常常见。这样我们可以将我们程序分摊到多个计算机CPU中去计算,在
基本概念: 浅拷贝:指对象的字段被拷贝,而字段引用的对象不会被拷贝,拷贝对象和原对象仅仅是引用名称有所不同,但是它们共用一份实体。对任何一个对象的改变,都会影响到另外一个对象。大部分的引用类型,实
.NET将原来独立的API和SDK合并到一个框架中,这对于程序开发人员非常有利。它将CryptoAPI改编进.NET的System.Security.Cryptography名字空间,使密码服务摆脱
文件与文件流的区别(自己的话): 在软件开发过程中,我们常常把文件的 “读写操作” ,与 “创造、移动、复制、删除操作” 区分开来
1. 前言 单元测试一直都是"好处大家都知道很多,但是因为种种原因没有实施起来"的一个老大难问题。具体是否应该落地单元测试,以及落地的程度, 每个项目都有自己的情况。 本篇为
事件处理 1、事件源:任何一个HTML元素(节点),body、div、button 2、事件:你的操作 &
1、什么是反射? 反射 (Reflection) 是 Java 的特征之一,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性。 Oracle 官方对
1、源码展示 ? 1
Java 通过JDBC获得连接以后,得到一个Connection 对象,可以从这个对象获得有关数据库管理系统的各种信息,包括数据库中的各个表,表中的各个列,数据类型,触发器,存储过程等各方面的信息。
可能大家谈到反射面部肌肉都开始抽搐了吧!因为在托管语言里面,最臭名昭著的就是反射!它的性能实在是太低了,甚至在很多时候让我们无法忍受。不过不用那么纠结了,老陈今天就来分享一下如何来优化反射!&nbs
1. 前言 最近一段时间一直在研究windows 驱动开发,简单聊聊。 对比 linux,windows 驱动无论是市面上的书籍,视频还是社区,博文以及号主,写的人很少,导
问题:ifndef/define/endif”主要目的是防止头文件的重复包含和编译 ========================================================
不知不觉.Net Core已经推出到3.1了,大多数以.Net为技术栈的公司也开始逐步的切换到了Core,从业也快3年多了,一直坚持着.不管环境
以前面试的时候经常会碰到这样的问题.,叫你写一下ArrayList.LinkedList.Vector三者之间的区别与联系:原先一直搞不明白,不知道这三者之间到底有什么区别?哎,惭愧,基础太差啊,木
目录 @RequestParam(required = true)的误区 先说结论 参数总结 @RequestParam(r
目录 FTP、FTPS 与 SFTP 简介 FTP FTPS SFTP FTP 软件的主动模式和被动模式的区别
1、Visitor Pattern 访问者模式是一种行为模式,允许任意的分离的访问者能够在管理者控制下访问所管理的元素。访问者不能改变对象的定义(但这并不是强制性的,你可以约定为允许改变)。对管
我是一名优秀的程序员,十分优秀!