Java实例化对象过程中的内存分配:

https://blog.csdn.net/qq_36934826/article/details/82685791

问题引入这里先定义一个很不标准的“书”类,这里为了方便演示就不对类的属性进行封装了。
class Book{    String name;    //书名    double price;   //价格    public void getInfo(){        System.out.println("name:"+name+";price:"+price);    }}1问题引入这里先定义一个很不标准的“书”类,这里为了方便演示就不对类的属性进行封装了。
class Book{    String name;    //书名    double price;   //价格    public void getInfo(){        System.out.println("name:"+name+";price:"+price);    }}1234567在这个类中定义了两个属性和一个方法,当然也是可以定义多和类和多个方法的。 类现在虽然已经定义好了,但是一个类要使用它必须要实例化对象,那么对象的定义格式有一下两种格式:
//声明并实例化对象: 类名称 对象名称 = new 类名称()Book book = new Book();123//分步完成声明和实例操作: // |- 声明对象: 类名称 对象名称 = null;Book book = null;// |- 实例化对象: 对象名称 = new 类名称();book = new Book();12345对象属于引用数据类型,其和基本数据类型最大的不同在于引用数据类型需要进行内存分配,而关键字new主要的功能就是开辟内存空间,也就是说只要是使用引用数据类型就必须使用关键字new来开辟空间。有些时候我们需要对对象属性进行操作,那么其中的堆栈内存空间又是如何分配的呢?接下来我们来分析一下其中的过程。
堆内存与栈内存如果想对对象操作的过程进行内存分析,首先要了解两块内存空间的概念:
堆内存:保存每一个对象的属性内容,堆内存需要用关键字new才能开辟。栈内存:保存的是一块堆内存的地址。堆内存很好理解,可能有人会有疑问为什么会有栈内存,举个例子,好比学校有很多教室,每个教室有一个门牌号,教室内放了很多的桌椅等等,这个编号就好比地址,老师叫小明去一个教室拿东西,老师必须把房间号告诉小明才能拿到,也就是为什么地址必须存放在一个地方,而这个地方在计算机中就是栈内存。
对象空属性我们先实例化一个对象,并对其的属性不设置任何值
public class Test{    public static void main(String args[]){         Book book = new Book();         book.getInfo();    }}123456运行结果如下:
name:null;price:0.01其内存变化图如下:

使用关键字new就在栈内存中开辟一个空间存放book对象,并且指向堆内存的一个空间,此时并未对其赋值,所以始终指向默认的堆内存空间。
操作对象属性我们先声明并实例化Book类,并对实例出的book对象操作其属性内容。
public class Test{    public static void main(String args[]){         Book book = new Book();         book.name = "深入理解JVM";         book.price = 99.8;         book.getInfo();    }}12345678编译执行后的结果如下:
name:深入理解JVM;price:99.81内存变化图如下:

分步实例化对象示例代码如下:
public class Test{    public static void main(String args[]){         Book book = null;  //声明对象         book = new Book(); //实例化对象         book.name = "深入理解JVM";         book.price = 99.8;         book.getInfo();    }}123456789很明显结果肯定和前面一样
name:深入理解JVM;price:99.81表面没什么区别,但是内存分配过程却不一样,接下来我们来分析一下

任何情况下只要使用了new就一定要开辟新的堆内存空间,一旦堆内存空间开辟了,里面就一定会所有类中定义的属性内容,此时所有的属性内容都是其对应数据类型的默认值。
直观的说就是栈内存先要指向一个null,然后等待开辟新的栈内存空间后才能指向其属性内容。
NullPointerException的出现那么如果使用了没有实例化的对象,就会出现最常见也是最让人头疼的一个异常NullPointerException,像下面的代码
public class Test{    public static void main(String args[]){         Book book = null;//         book = new Book();   //实例化的这一步被注释         book.name = "深入理解JVM";         book.price = 99.8;         book.getInfo();    }}123456789在编译的过程是不会出错的,因为只有语法错误才会在编译时中断,而这种逻辑性错误能成功编译,但是执行的时候却会抛出NullPointerException异常。 运行结果:
Exception in thread "main" java.lang.NullPointerException at language.Test.main(Test.java:19)1空指针异常是平时遇到最多的一类异常,只要是引用数据类型都有可能出现它。这种异常的出现也是很容易理解的,犹如你说今天被一只恐龙追着跑,恐龙早就在几个世纪前就灭绝了,现实生活中不可能存在,当然人们就会认为你说的这句话是谎言。在程序中也一样,没有被实例化的对象直接调用其中的属性或者方法,肯定会报错。
引用数据分析引用是整个java中的核心精髓,引用类似于C++中的指针概念,但是又比指针的概念更加简单。 举个简单的例子,比如李华的小名叫小华,一天李华因为生病向老师请假了,老师问今天谁请假了,说李华请假了和小华请假了都是一个意思,小华是李华的别名,他们两个都是对应一个个体。 如果代码里面声明两个对象,并且使用了关键字new为两个对象分别进行了对象的实例化操作,那么一定是各自占用各自的堆内存空间,并且不会互相影响。
例如:声明两个对象
public class Test{    public static void main(String args[]){         Book bookA = new Book();         Book bookB = new Book();
         bookA.name = "深入理解JVM";         bookA.price = 99.8;         bookA.getInfo();
         bookB.name = "Java多线程";         bookB.price = 69.8;         bookB.getInfo();    }}1234567891011121314运行结果如下:
name:深入理解JVM;price:99.8name:Java多线程;price:69.812我们来分析一下内存的变化

接下来我们看看那对象引用传递
例如:对象引用传递
public class Test{    public static void main(String args[]){         Book bookA = new Book();   //声明并实例化对象         Book bookB = null;         //声明对象         bookA.name = "深入理解JVM";         bookA.price = 99.8;         bookB = bookA;             //引用传递         bookB.price = 69.8;         bookA.getInfo();    }}1234567891011运行结果如下:
name:深入理解JVM;price:69.81严格来讲bookA和bookB里面保存的是对象的地址信息,所以以上的引用过程就属于将bookA的地址赋给了bookB,此时两个对象指向的是同一块堆内存空间,因此任何一个对象修改了堆内存之后都会影响其他对象。

一块堆内存可以同时被多个栈内存所指向,但是反过来,一块栈内存只能保存一块堆内存空间的地址。
垃圾的产生先看如下代码:
public class Test{    public static void main(String args[]){         Book bookA = new Book();   //声明并实例化对象         Book bookB = new Book();   //声明并实例化对象         bookA.name = "深入理解JVM";         bookA.price = 99.8;         bookB.name = "Java多线程";         bookB.price = 69.8;         bookB = bookA;             //引用关系         bookB.price = 120.8;         bookA.getInfo();    }}12345678910111213运行结果如下:
name:深入理解JVM;price:120.81整个过程内存又发生了什么变化呢?我们来看一下 
在此过程中原来bookB所指向的堆内存无栈内存指向,一块没有任何栈内存指向的堆内存空间就将成为垃圾,等待被java中的回收机制回收,回收之后会释放掉其占用的空间。
虽然在java中支持了自动的垃圾收集处理,但是在代码的编写过程中应该尽量减少垃圾空间的产生。
欢迎来我的博客网站逛一逛~blog.beifengtz.com--------------------- 作者:beifengtz 来源:CSDN 原文:https://blog.csdn.net/qq_36934826/article/details/82685791 版权声明:本文为博主原创文章,转载请附上博文链接!234567在这个类中定义了两个属性和一个方法,当然也是可以定义多和类和多个方法的。 类现在虽然已经定义好了,但是一个类要使用它必须要实例化对象,那么对象的定义格式有一下两种格式:
//声明并实例化对象: 类名称 对象名称 = new 类名称()Book book = new Book();123//分步完成声明和实例操作: // |- 声明对象: 类名称 对象名称 = null;Book book = null;// |- 实例化对象: 对象名称 = new 类名称();book = new Book();12345对象属于引用数据类型,其和基本数据类型最大的不同在于引用数据类型需要进行内存分配,而关键字new主要的功能就是开辟内存空间,也就是说只要是使用引用数据类型就必须使用关键字new来开辟空间。有些时候我们需要对对象属性进行操作,那么其中的堆栈内存空间又是如何分配的呢?接下来我们来分析一下其中的过程。
堆内存与栈内存如果想对对象操作的过程进行内存分析,首先要了解两块内存空间的概念:
堆内存:保存每一个对象的属性内容,堆内存需要用关键字new才能开辟。栈内存:保存的是一块堆内存的地址。堆内存很好理解,可能有人会有疑问为什么会有栈内存,举个例子,好比学校有很多教室,每个教室有一个门牌号,教室内放了很多的桌椅等等,这个编号就好比地址,老师叫小明去一个教室拿东西,老师必须把房间号告诉小明才能拿到,也就是为什么地址必须存放在一个地方,而这个地方在计算机中就是栈内存。
对象空属性我们先实例化一个对象,并对其的属性不设置任何值
public class Test{    public static void main(String args[]){         Book book = new Book();         book.getInfo();    }}123456运行结果如下:
name:null;price:0.01其内存变化图如下:

使用关键字new就在栈内存中开辟一个空间存放book对象,并且指向堆内存的一个空间,此时并未对其赋值,所以始终指向默认的堆内存空间。
操作对象属性我们先声明并实例化Book类,并对实例出的book对象操作其属性内容。
public class Test{    public static void main(String args[]){         Book book = new Book();         book.name = "深入理解JVM";         book.price = 99.8;         book.getInfo();    }}12345678编译执行后的结果如下:
name:深入理解JVM;price:99.81内存变化图如下:

分步实例化对象示例代码如下:
public class Test{    public static void main(String args[]){         Book book = null;  //声明对象         book = new Book(); //实例化对象         book.name = "深入理解JVM";         book.price = 99.8;         book.getInfo();    }}123456789很明显结果肯定和前面一样
name:深入理解JVM;price:99.81表面没什么区别,但是内存分配过程却不一样,接下来我们来分析一下

任何情况下只要使用了new就一定要开辟新的堆内存空间,一旦堆内存空间开辟了,里面就一定会所有类中定义的属性内容,此时所有的属性内容都是其对应数据类型的默认值。
直观的说就是栈内存先要指向一个null,然后等待开辟新的栈内存空间后才能指向其属性内容。
NullPointerException的出现那么如果使用了没有实例化的对象,就会出现最常见也是最让人头疼的一个异常NullPointerException,像下面的代码
public class Test{    public static void main(String args[]){         Book book = null;//         book = new Book();   //实例化的这一步被注释         book.name = "深入理解JVM";         book.price = 99.8;         book.getInfo();    }}123456789在编译的过程是不会出错的,因为只有语法错误才会在编译时中断,而这种逻辑性错误能成功编译,但是执行的时候却会抛出NullPointerException异常。 运行结果:
Exception in thread "main" java.lang.NullPointerException at language.Test.main(Test.java:19)1空指针异常是平时遇到最多的一类异常,只要是引用数据类型都有可能出现它。这种异常的出现也是很容易理解的,犹如你说今天被一只恐龙追着跑,恐龙早就在几个世纪前就灭绝了,现实生活中不可能存在,当然人们就会认为你说的这句话是谎言。在程序中也一样,没有被实例化的对象直接调用其中的属性或者方法,肯定会报错。
引用数据分析引用是整个java中的核心精髓,引用类似于C++中的指针概念,但是又比指针的概念更加简单。 举个简单的例子,比如李华的小名叫小华,一天李华因为生病向老师请假了,老师问今天谁请假了,说李华请假了和小华请假了都是一个意思,小华是李华的别名,他们两个都是对应一个个体。 如果代码里面声明两个对象,并且使用了关键字new为两个对象分别进行了对象的实例化操作,那么一定是各自占用各自的堆内存空间,并且不会互相影响。
例如:声明两个对象
public class Test{    public static void main(String args[]){         Book bookA = new Book();         Book bookB = new Book();
         bookA.name = "深入理解JVM";         bookA.price = 99.8;         bookA.getInfo();
         bookB.name = "Java多线程";         bookB.price = 69.8;         bookB.getInfo();    }}1234567891011121314运行结果如下:
name:深入理解JVM;price:99.8name:Java多线程;price:69.812我们来分析一下内存的变化

接下来我们看看那对象引用传递
例如:对象引用传递
public class Test{    public static void main(String args[]){         Book bookA = new Book();   //声明并实例化对象         Book bookB = null;         //声明对象         bookA.name = "深入理解JVM";         bookA.price = 99.8;         bookB = bookA;             //引用传递         bookB.price = 69.8;         bookA.getInfo();    }}1234567891011运行结果如下:
name:深入理解JVM;price:69.81严格来讲bookA和bookB里面保存的是对象的地址信息,所以以上的引用过程就属于将bookA的地址赋给了bookB,此时两个对象指向的是同一块堆内存空间,因此任何一个对象修改了堆内存之后都会影响其他对象。

一块堆内存可以同时被多个栈内存所指向,但是反过来,一块栈内存只能保存一块堆内存空间的地址。
垃圾的产生先看如下代码:
public class Test{    public static void main(String args[]){         Book bookA = new Book();   //声明并实例化对象         Book bookB = new Book();   //声明并实例化对象         bookA.name = "深入理解JVM";         bookA.price = 99.8;         bookB.name = "Java多线程";         bookB.price = 69.8;         bookB = bookA;             //引用关系         bookB.price = 120.8;         bookA.getInfo();    }}12345678910111213运行结果如下:
name:深入理解JVM;price:120.81整个过程内存又发生了什么变化呢?我们来看一下 
在此过程中原来bookB所指向的堆内存无栈内存指向,一块没有任何栈内存指向的堆内存空间就将成为垃圾,等待被java中的回收机制回收,回收之后会释放掉其占用的空间。
虽然在java中支持了自动的垃圾收集处理,但是在代码的编写过程中应该尽量减少垃圾空间的产生。--------------------- 作者:beifengtz 来源:CSDN 原文:https://blog.csdn.net/qq_36934826/article/details/82685791 版权声明:本文为博主原创文章,转载请附上博文链接!

Java实例化对象过程中的内存分配的更多相关文章

  1. java中子类实例化过程中的内存分配

    知识点: 子类继承父类之后,实例化子类时,内存中子类是如何分配内存的呢? 下面,自己会结合一个例子,解释一下,一个子类实例化过程中,内存是如何分配的 参考博客:http://www.cnblogs.c ...

  2. Java 程序运行过程中的内存分析

    作为 java 程序员,都应该知道 Java 程序运行在 JVM(Java Virtual Machine,Java 虚拟机)上,可以把 JVM 理解成 Java 程序和操作系统之间的桥梁,JVM 实 ...

  3. Java中对象创建时的内存分配

    一.前言知识铺垫   1.逃逸对象:在一个方法内创建的对象没有被外界引用则称该对象为未逃逸的对象. 2.JDK1.6以后的HotSpot虚拟机支持运行时的对象逃逸分析. 3.JVM中的参数配置: 1) ...

  4. Java中类,对象,方法的内存分配

    Java中类,对象,方法的内存分配 以下针对引用数据类型: 在内存中,类是静态的概念,它存在于内存中的CodeSegment中. 当我们使用new关键字生成对象时,JVM根据类的代码,去堆内存中开辟一 ...

  5. Java中的内存分配机制

    Java的内存分为两种:一种是栈内存,一种是堆内存. 在函数中定义的一些基本类型变量和对象的引用都在函数的栈内存中分配.当在一个代码块中定义一个变量的时候,java就在栈中为其分配内存,当超过作用域的 ...

  6. Java虚拟机中的内存分配

    java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途以及创建和销毁的时间. 栈:存放的是局部变量,包括:1.用来保存基本数据类型的值:2.保存类 ...

  7. Java基础-Java中的内存分配与回收机制

    Java基础-Java中的内存分配与回收机制 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一. 二.

  8. 《深入理解Java虚拟机》学习笔记之内存分配

    JVM在执行Java程序的过程中会把它所管理的内存划分若干个不同的数据区域,如下图: 大致可以分为两类:线程私有区域和线程共享区域. 线程私有区域 程序计数器(Program Counter Regi ...

  9. Netty 中的内存分配浅析

    Netty 出发点作为一款高性能的 RPC 框架必然涉及到频繁的内存分配销毁操作,如果是在堆上分配内存空间将会触发频繁的GC,JDK 在1.4之后提供的 NIO 也已经提供了直接直接分配堆外内存空间的 ...

随机推荐

  1. mysql 树结构递归处理

    日常开发中我们经常会遇到树形结构数据处理,一般表结构通常会常用id,pid这种设计方案. 之前用oracle.sqlServer数据库,用相应的语法即可获取树形结构数据(oracel:connect ...

  2. IP-reputation feature

    IP-reputation feature https://blog.norz.at/citrix-netscaler-ip-reputation-feature/ I recently had to ...

  3. 二、eureka服务端注册服务

    所有文章 https://www.cnblogs.com/lay2017/p/11908715.html 正文 入口 上文我们说到,eureka是使用jersey来对外提供restful风格的rpc调 ...

  4. Centos7 配置 svn服务端

    转载至:Linux(阿里云Centos7)环境下搭建svn服务器以及权限配置详细步骤 本篇文章主要介绍在CentOS7中采用yum安装方式.优点:简单,一键安装,不用手动配置环境变量等.缺点:安装位置 ...

  5. leetcode-102.层序遍历二叉树(正序)· BTree

    题面 Given a binary tree, return the level order traversal of its nodes' values. (ie, from left to rig ...

  6. How to find Oracle EBS Weblogic Server Admin Port and URL

    How to find Oracle EBS Weblogic Server Admin Port and URL   Weblogic admin portMethod 1 Open the App ...

  7. You're currently running Fcitx with GUI 错误解决 Fcitx

    在英文版ubuntu配置输入法时,点击 Configure Current Input Method 会报以下的错误: You’re currently running Fcitx with GUI, ...

  8. [Docker][Hadoop]基于Docker1.12.3 搭建Hadoop 2.7.2 集群以及简单分析

    一 Hadoop简介 Hadoop 2.7.2 Doc refer to http://hadoop.apache.org/docs/r2.7.2/ HDFS (The following is a ...

  9. Vs2017 FrameWork EF Mysql 控制台应用

    1  运行环境   vs2017   Net FromWork 4.6.2  手动版 没有 ado.net 实体数据模型 2 NuGet  MySql.Data.Entity 6.10.9, MySq ...

  10. k2系列-安装篇

    K2介绍: K2是基于BPM的流程开发平台,它支持在net开发环境/visio/moss等不同环境下进行流程开发. K2本身部署简单,操作灵活,非常适合大中型企业流程开发和部署. K2安装步骤: 首先 ...