在讲述这些之前我们需要一些预备知识:

Java的内存结构我们可以通过两个方面去看待它。

从该角度看的话Java内存结构包含以下部分:该部分内容可以结合:JVM简介(更加详细深入的介绍)

1、栈区:由编译器自动分配释放,具体方法执行结束后,系统自动释放JVM内存资源。

其作用有保存局部变量的值,包括:1.用来保存基本数据类型的值;2.保存类的实例对象的引用。也可以用来保存加载方法时的帧。

2、堆区:一般由程序员分配释放,JVM不定时查看这个对象,如果没有引用指向这个对象就回收。

其作用为用来存放动态产生的数据,包括new出来的实例,数组等。注意创建出来的对象只包含属于各自的成员变量,并不包括成员方法。

因为同一个类的对象拥有各自的成员变量,存储在各自的堆中,但是他们共享该类的方法,并不是每创建一个对象就把成员方法复制一次。

3、代码区:存放程序中方法的二进制代码,而且是多个对象共享一个代码空间区域。

4、数据区:用来存放static定义的静态成员。

5、常量池:JVM为每个已加载的类型维护一个常量池,常量池就是这个类型用到的常量的一个有序集合。包括直接常量(基本类型,String)和对其他类型、方法、字段的符号引用。池中的数据和数组一样通过索引访问。由于常量池包含了一个类型所有的对其他类型、方法、字段的符号引用,所以常量池在Java的动态链接中起了核心作用。常量池存在于堆中。

下图大致描述了JAVA的内存分配

二、从操作系统上的进程的角度。相关定义请参考各种操作系统的资料,例如Linux的话可以参考这个简单的介绍:Linux Processes explained  (此方面一般被较少地谈论到,本文对此仅仅做一个稍微的介绍)

这里切记一点:JVM规范所描述的抽象JVM概念与实际实现并不总一一对应。

接来下我们来看一段代码实例与注释:

public classTestStringConstant {public static voidmain(String args[]) {//字符串常量,分配在常量池中,编译器会对其进行优化, Interned table//即当一个字符串已经存在时,不再重复创建一个相同的对象,而是直接将s2也指向"hello".

String s1 = "hello";

String s2= "hello";//new出来的对象,分配在heap中.s3与s4虽然它们指向的字符串内容是相同的,但是是两个不同的对象.//因此==进行比较时,其所存的引用是不同的,故不会相等

String s3 = new String("world");

String s4= new String("world");

System.out.println(s1== s2); //true

System.out.println(s3 == s4); //false

System.out.println(s3.equals(s4)); //true//String中equals方法已经被重写过,比较的是内容是否相等.

}

}

那么对于上例代码中提到的编译器的优化,下面将进行更进一步的详细介绍。请看下例代码:

classA {private String a = "aa";public booleanmethodB() {

String b= "bb";final String c = "cc";return false;

}

}

"aa"、"bb"的String对象按JVM规范在Java heap上,在JDK8之前的HotSpot VM实现里在PermGen,在JDK7开始的HotSpot VM里在普通Java heap里(而不在PermGen里);"cc"如果存在的话也一样,但是可能会不存在。

这些String对象属于“interned String”。String是Java对象,根据JVM规范的定义它必须存在于Java heap中,interned String也不例外。Interned String特别的地方在于JVM会有个StringTable存着interned String的引用,保证内容相同的String对象不被重复intern。(这里便是编译器的优化)

这个StringTable怎样实现JVM规范里并没有规定,不过通常它并不保存String对象的内容,而只是保存String对象的引用而已。

从JVM规范看a、b、c变量:

a变量作为A类的对象实例字段,会跟随A的实例在Java heap上。

b变量作为局部变量会在Java线程栈上。

c变量虽然也是局部变量,但因为有final修饰并且有初始化为一个常量值,所以c是一个常量。它可能会被优化掉(就没有c这个变量了),也可能跟b一样作为局部变量在Java线程栈上。

通过以上相信大家对于字符串常量的分配区域以及java的内存分配有了一个较为形象的了解。

下面是一些相关知识点的补充与注意事项:

1.分清什么是实例什么是对象。Class a= new Class();此时a叫实例,而不能说a是对象。实例在栈中,对象在堆中,操作实例实际上是通过实例的指针间接操作对象。多个实例可以指向同一个对象。

2.栈中的数据和堆中的数据销毁并不是同步的。方法一旦结束,栈中的局部变量立即销毁,但是堆中对象不一定销毁。因为可能有其他变量也指向了这个对象,直到栈中没有变量指向堆中的对象时,它才销毁,而且还不是马上销毁,要等垃圾回收扫描时才可以被销毁。

3.以上的栈、堆、代码段、数据段等等都是相对于应用程序而言的。每一个应用程序都对应唯一的一个JVM实例,每一个JVM实例都有自己的内存区域,互不影响。并且这些内存区域是所有线程共享的。这里提到的栈和堆都是整体上的概念,这些堆栈还可以细分。

4.类的成员变量在不同对象中各不相同,都有自己的存储空间(成员变量在堆中的对象中)。而类的方法却是该类的所有对象共享的,只有一套,对象使用方法的时候方法才被压入栈,方法不使用则不占用内存。

从HotSpot VM的实现看:

当methodB()被解释执行时,输入的字节码是怎样的就会怎样执行,而由于javac的实现不会优化掉变量b,所以调用methodB()时它一定会在Java线程栈上的局部变量区里;当字节码里变量c存在时,它也跟b一样在Java线程栈的局部变量区。

当methodB()被JIT编译执行时,由于局部变量b、c都没有被使用,所以它们经过JIT编译后就消失了,调用methodB()不会在栈上给b或c变量分配任何空间。

通过以上相信大家对于字符串常量的分配区域以及java的内存分配有了一个较为形象的了解。

下面是一些相关知识点的补充与注意事项:

1.分清什么是实例什么是对象。Class a= new Class();此时a叫实例,而不能说a是对象。实例在栈中,对象在堆中,操作实例实际上是通过实例的指针间接操作对象。多个实例可以指向同一个对象。

2.栈中的数据和堆中的数据销毁并不是同步的。方法一旦结束,栈中的局部变量立即销毁,但是堆中对象不一定销毁。因为可能有其他变量也指向了这个对象,直到栈中没有变量指向堆中的对象时,它才销毁,而且还不是马上销毁,要等垃圾回收扫描时才可以被销毁。

3.以上的栈、堆、代码段、数据段等等都是相对于应用程序而言的。每一个应用程序都对应唯一的一个JVM实例,每一个JVM实例都有自己的内存区域,互不影响。并且这些内存区域是所有线程共享的。这里提到的栈和堆都是整体上的概念,这些堆栈还可以细分。

4.类的成员变量在不同对象中各不相同,都有自己的存储空间(成员变量在堆中的对象中)。而类的方法却是该类的所有对象共享的,只有一套,对象使用方法的时候方法才被压入栈,方法不使用则不占用内存。

对于String的相关补充:

对于String的修改其实是new了一个StringBuilder并调用append方法,然后调用toString返回一个新的String.

(注意:append方法并不会new一个新的对象.)

但是JVM是会对String进行优化的,比如:

String str = "I" + "love" + "java"

其中的字符串在编译的时候就能够确认,所以编译器会直接将其拼接成一个字符串放在常量池:"I love java"

但是若代码为下面这样:

String a = "I";

String b = "love";

String c = "java";

String str = a + b + c;

那么只有等到运行的时候才能够确定str最终是什么,编译器并不会对其进行优化,而是通过StringBuilder对字符串改变来实现的。

但是注意,要是此处给 a, b, c添加上 final 关键字,则编译器就能够对其进行优化。我们可以做下面这样一个测试:

public classfoo{public static voidmain(String[] args) {

String a= "I ";

String b= "love ";

String c= "java";final String a1 = "I ";final String b1 = "love ";final String c1 = "java";

String str= a + b +c;

String str1= a1 + b1 + c1; //equals to str1 = "I " + "love " + "java"

String str2 = "I " + "love " + "java";

System.out.println(str== "I love java"); //output false

System.out.println(str1 == "I love java"); //output true

System.out.println(str2 == "I love java"); //output true

System.out.println(a + "love " + "java" == "I love java"); //output false

System.out.println(a1 + "love " + "java" == "I love java"); //output true

}

}

Note:

StringBuilder 和 StringBuffer 的区别:

StringBuffer 是线程安全的;StringBuilder 是非线程安全的。

因为StingBuffer是在StringBuilder的基础上加锁,而加锁是一个重量级的操作,需要调用操作系统内核来实现。

比较耗时。

故在效率上: StringBuilder > StringBuffer

java字符串常量存哪里_浅谈JAVA中字符串常量的储存位置相关推荐

  1. java中修饰常量的事_浅谈java中的声明常量为什么要用static修饰

    今天定义一个类常量,想着也只有这个类可以用到,就没用static关键字修饰.结果sonar代码检查提示: Rename this field "PERSON_TYPE_USER" ...

  2. java方法区对象类型_浅谈Java内存区域与对象创建过程

    一.java内存区域 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有的区域则 ...

  3. java中单例的应用_浅谈Java中单例模式的几种应用

    目录 浅谈Java中单例模式的几种应用 第一种:懒汉式 第二种:饿汉式 第三种:双重检索式 第四种:注册登记式 第五种:内部类形式 浅谈Java中单例模式的几种应用 日常开发中,为了提高我们系统中对象 ...

  4. java的向下转型_浅谈Java向下转型的意义

    一开始学习 Java 时不重视向下转型.一直搞不清楚向下转型的意义和用途,不清楚其实就是不会,那开发的过程肯定也想不到用向下转型. 其实向上转型和向下转型都是很重要的,可能我们平时见向上转型多一点,向 ...

  5. php和java的区别菜鸟教程_浅谈Java和PHP的主要区别

    当谈到PHP与Java的差异性问题时,更多的是回答初学者的一些疑问.对于刚接触IT的同学来说,他们需要做好对未来职业的选择.所以是选择PHP还是选择Java更有利于自身的技术特点和发展前景.所以在解决 ...

  6. java 常量放在哪里_浅谈JAVA中字符串常量的储存位置

    在讲述这些之前我们需要一些预备知识: Java的内存结构我们可以通过两个方面去看待它. 从该角度看的话Java内存结构包含以下部分: 1.栈区:由编译器自动分配释放,具体方法执行结束后,系统自动释放J ...

  7. java类的命名规范_浅谈Java中的命名规范

    现代软件架构的复杂性需要协同开发完成,如何高效地协同呢? 答案是:制定一整套统一的规范. 无规矩不成方圆,无规范难以协同,比如,制订交通法规表面上是要限制行车权,实际上是保障公众的人身安全,试想如果没 ...

  8. java接口是干啥_浅谈Java接口

    接口(英文:Interface)是Java中非常重要的内容,初学的时候可能感受不深,但是在做项目的时候,对面向接口编程的运用就变得尤为重要,不过这是后话了.现在先讨论假如是刚刚接触接口这个概念,该怎么 ...

  9. java四种内部类区别_浅谈Java中的四种内部类

    如果你看过一些JDK和框架源码的话,就经常会发现一般在类的定义中,都会再定义一些其他的类,这些类也同样会被编译成字节码文件,这样的类就被叫做 内部类 ,按照一般的分法,大致可以分为以下四类: 成员内部 ...

最新文章

  1. strut2服务器与android交互数据,用Android搭建客户端 手机和服务器交互开发
  2. K8S Runtime CRI OCI contained dockershim 理解(转)
  3. Docker的镜像导出与导入与拷贝
  4. 安装最新Spree出现error:spree_core requires will_paginate (= 3.0.pre2, runtime)
  5. 俞军:为什么多数产品经理都不合格
  6. DevOps通用及版本控制面试题
  7. 飞鸽传书内部排序算法的性能比较
  8. Box2D教程5-碰撞检测
  9. python查看指令的方法python -h
  10. c语言 分函数,C语言部分函数.doc
  11. [bzoj1044][HAOI2008]木棍分割
  12. 安卓Behavior用法
  13. 从小白到Python大神只需要100天
  14. 凝思系统激活序列号_凝思磐石安全操作系统
  15. 华三模拟器启动设备失败【启动设备MSR36-20_1失败】
  16. 倒数日怎么显示在桌面_偏头痛是怎么来的?医生:和这4个因素关系密切,要注意...
  17. 学习安卓的简单心得,以及LinearLayout的简单使用
  18. 4G、5G RLC AM模式数据和控制报文格式的区别
  19. Anchor Based和Anchor Free的相爱相杀与ATSS的诞生
  20. C语言 主动判别int型出界

热门文章

  1. matlab的knn均值滤波,中值滤波与均值滤波介绍.ppt
  2. 计算机xp的解释,2017职称计算机考点:Windowsxp系统注册表的技巧
  3. linux安装mathematica7,Linux下面如何安装maple,mathematica,matlab这类软件?
  4. 大唐豪侠服务器列表文件格式错误,[动态] 大唐豪侠1.2.4版本更新公告-大唐豪侠-东北游戏网...
  5. html 点击子元素,html如何点击子元素事件而不触发父元素的点击事件——阻止冒泡...
  6. 玩转 SpringBoot 2 快速整合 | RESTful Api 篇
  7. 基于JAVA+SpringMVC+Mybatis+MYSQL的同学录管理系统
  8. 基于JAVA+Servlet+JSP+MYSQL的图书销售管理系统
  9. 广东中国科学院计算机网络信息中心,中国科学院
  10. 【题解】luoguP2680运输计划