Java 面向对象通过new揭开对象实例化

网友投稿 410 2022-08-06


Java 面向对象通过new揭开对象实例化

目录概念了解第一种情况第二种情况

更正说明:

我之前的的标题有点文不对题,我这篇博客的内容明明说的是:java中对象创建的过程,对内存之种底层的东西,我其实提的不太多。

所以我原来的标题:《当你实例化对象时,在内存中到底发生了什么》是不合适的

概念了解

想要知道内存中发生了什么,我们先来了解两个内存概念

堆内存:保存对象的属性内容。堆内存需要用new关键字来分配空间;栈内存:保存的是堆内存的地址(在这里为了分析方便,可以简单理解为栈内存保存的是对象的名字)。

Student student1 = new Student()

这行代码可以分为两部分

1、Student student1 声明对象,会在栈上开辟一块区域,名叫student1,里面存了:堆中开辟的内存的地址

2、new Student() 实例化student1对象,在堆里开辟了一块区域,名叫student1

当用等号将两者关联的时候,其实就是将栈上声明的对象student1,指向堆上实例化的具体空间

如图所示:

从图中也可以看出,在“=”关联的时候,就是将堆中student1开辟内存的地址0X99存放到了栈里,可以吧内存地址看成Student类的一把钥匙

实例化后对象student1获得了这把钥匙,所以可以在类中任意拿东西,即调用属性、方法等

(大多数情况下可以这么认为,其实变量名student1储存的不一定是地址)

第一种情况

因此可以通过对象名 student1 ,给堆中的具体成员变量赋值、调用。

其实栈中存放的就是堆中某一个内存的引用,这就是对象引用的概念,其中的 student1 也被称作对象的引用变量

class Student {

String name; //学生类的属性

int age;

public void eat() {

System.out.println(name + "在吃饭, 他" + age + "岁了"); // 学生类的行为(方法)

}

}

public class TestDemo3 {

// 不同的实例化对象在栈中拥有不同的空间,他们也指向堆中不同的内存地址

public static void main(String[] args) {

Student student1 = new Student(); // 学生对象student1

student1.name = "小强"; // 通过 对象名.属性 来给堆中的具体成员变量赋值

student1.age = 18;

student1.eat(); // 调用类中的eat()方法

Student student2 = new Student(); // 学生对象student2

student2.name = "小红";

student2.age = 23;

student2.eat();

}

}

第二种情况

不同的实例化对象在栈中拥有不同的空间,他们也指向堆中不同的内存地址,但是也可以有这样的情况:栈中有不同的空间,他们指向堆中同一个内存地址。

class Dog {

String name;

int age;

public void show() {

System.out.println("姓名:" + name + " 年龄: " + age);

}

}

public class TestDemo4 {

public static void main(String[] args) {

Dog one = new Dog();

Dog two = one;

}

}

对象one在实例化后,拥有一块栈的空间,空间里存放堆的一个内存地址

那么将one赋值给two之后,two的栈的空间里,也会存放one拥有的堆的内存地址。

用一个很形象的比喻,妻子带着一幅画嫁给了丈夫,那么她的画也就属于丈夫了,两人共有。

不论是丈夫还是妻子,只要是对画做出了最后的更改,那么这幅画的内容就会改变。即最后一次修改是有效的,会覆盖掉原先的内容

用代码来展示:

class Dog {

String name;

int age;

public void show() {

System.out.println("姓名:" + name + " 年龄: " + age);

}

}

public class TestDemo4 {

public static void main(String[] args) {

Dog one = new Dog();

Dog two = one;

one.name = "阿黄";

one.age = 10;

two.name = "旺财";

two.age = 3;

one.show();

System.out.println("========这是分割线========");

two.show();

}

}

怎么样!通过对象one对姓名和年龄的赋值是不是被覆盖掉了,内存中的场景大概是这样

栈中的对象one、two同时指向堆中的同一块空间,也就是不同的栈可以修改同一块堆内存的内容,这样堆内存的内容自然就是最后对象two所赋值的内容了。

哈哈! 原来在我们实例化new一个对象的时候,内存里偷偷干了这么多事啊!

各位铁汁们,我们下篇博客见,嘻嘻


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:设计人员和开发人员如何协作,如何与开发团队高效协作?
下一篇:idea创建maven父子工程导致子工程无法导入父工程依赖(idea导入maven包没引入进来)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~