JVM内存分配及String常用方法

一,JVM内存分配和常量池

? 在介绍String类之前,先来简单分析一下在JVM中,对内存的使用是如何进行分配的。如下图所示(注意:在jdk1.8之后便没有方法区了)

?

? 如上JVM将内存分为多个不同的区域,这些区域都有各自的用途、创建和销毁的时间,有些区域随虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束来建立和销毁。

? 区域名称的说明:

1.1,方法区:

? 属于数据共享内存区域,存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

1.2,虚拟机栈

? 虚拟机栈就是我们通常说的栈,是Java执行方法的内存模型,每当执行一次方法时,都会创建一个栈帧。把栈帧压入栈,当Java方法调用时返回正常的结果或者捕获异常时,栈帧出栈。

? 栈帧:栈帧存储方法的相关信息,包含局部变量数表、返回值、操作数栈、动态链接。

1.3,本地方法栈

? 从功能上来说与虚拟机栈类似,但是虚拟机栈执行的是字节码,而本地方法栈调用的是Native方法,并且它是线程独享的。

1.4,程序计数器

? 程序计数器是线程独享的,它是记录当前线程执行的字节码行号。在多线程执行时,CPU会来回在线程之间进行切换,那么当再次回到一条线程时,是如何得知线程的存储单元及执行指令。而程序计数器便会进行存储下一条存储单元的地址,执行完毕后程序计数器自动加 1 ,以此循环直到程序结束为止。

1.5,堆

? 说到堆这个概念想必都不陌生,它是内存中的重要角色。它主要是用来存储被创建出来的对象,通过关键字new实例出来的,是所有线程共享的一块最大的区域。

? ==特别注意:在JDK1.7及以后,常量池移动到堆内存中。==

? 堆还包括一个==常量池==,用来存储编译期间生成的==字面量和符号==引用。这部分内容在类被加载后,都会存储到方法区中。同时,运行时产生的新常量也可以被放入常量池中,比如 String 类中的 intern() 方法产生的常量。

? 常量池就是这个类型用到的常量的一个有序集合。包括直接常量(基本类型,String)对其他类型、方法、字段的符号引用。

二,常量池

2.1,什么是常量:

? 常量是指被final修饰的变量,值一旦确定就无法改变。

? final可以修饰静态变量、方法、实例变量和局部变量。

? 常量池分为两种形式:静态常量池和运行时常量池

2.2,静态常量池

? 即*.class文件中的常量池,class文件中的常量池不仅仅包含字符串(数字)字面量,还包含类、方法的信息,占用class文件绝大部分空间。这种常量池用于存放字面量和符号引用量。

2.3,运行时常量池

? 指JVM虚拟机在完成类装载操作后,将class文件中的常量池载入到内存中,并保存在方法区中,我们常说的常量池,就是指方法区中的运行时常量池。同样运行时常量池一个重要的特征就是具有动态性,指并不需要常量只有在编译期才会产生,在运行期也会将新的常量保存到常量池中,如String类中的intern()方法。

三,== 和equals

3.1,两者之间区边

? ==:

? 对于基本类型来说:==表示数值的比较

? 对于引用类型来说:==表示地址值的比较

? equals:

? 比较的是两者之间值是否相等,但是Java中的类都是直接或者间接继承Object类,而equals不也例外。其实在equals源码中也是使用==进行比较的,如下源码:

        ![](https://img2018.cnblogs.com/blog/1655301/201909/1655301-20190902223856542-1095893842.png)

? 那么问题来了,这和==又有什么区别呢?

? 上面说到equals也是继承自java.lang.Object,因此可以对equals进行重写来定义我们自己的比较方式。

? 请参看以下代码:

        String str1 = "abc";
        String str2 = "abc";

        char[] strArray = {'a','b','c'};
        String str3 = new String(strArray);

        String str4 = "abc";

        System.out.println(str1 == str2);
        System.out.println(str1 == str3);
        System.out.println(str2 == str3);
        System.out.println(str4.equals(str1)); 

? 以上运行结果为:

true
false
false
true

? 接下来我们依次分析上面的结果:

? 1,str1与str2比较的是字符串对象地址,因为它们的值是相同的,所以地址值也是相同的。

? 2,str3是new出来的示例对象,在堆内存中会开辟一块新的内存地址,它并不在常量池中。所以返回结果为false。

? 3,同理str2与str3比较也是一样的结果。

? 4,equals比较的是值是否相同,所以返回的结果为true。

如图所示:

?

四,String常用方法

? 首先声明字符串:

 String str1 = "abc";

4.1,int length()

 int length = str1.length();
 System.out.println(length);

4.2,char charAt(值)

String str= "abc";
char c = str.charAt(1);
System.out.println(c); 

4.3,char toCharArray()

String str= "abc";
char c[] = str.toCharArray();
for (int i = 0; i < c.length; i++) {
      System.out.println("转为数组输出:" + c[i]);
}

4.4,int indexOf("字符"); int lastIndexOf("字符")

String str="axcdefgabc";
int a1 = str.indexOf("a");
int a2 = str.indexOf("x",  2);
int a3 = str.lastIndexOf("c");
System.out.println("你的位置为:" + a1);
System.out.println("为的位置为:" + a2);
System.out.println("点最后出现的位置为:" + a3);

4.5,字符串大小写转换

? toUpperCase(); 转换成大写
? toLowerCase();转换成小写

String str = "hello world";
String str1 = "HELLO WORD";
System.out.println("将字符串转大写为:" + str.toUpperCase());
System.out.println("将字符串转换成小写为:" + str1.toLowerCase());

4.6,String[] split("字符")

String str = "abc,def,123";
String[] arr1 = str.split(",");

4.7,boolean equals(Object anObject)

 String str = "abc";
 String str1= "123";
 if(str.equals(str1)) {
      System.out.println("相等");
 }
 else{
      System.out.println("不相等");
 }

4.8,String trim()

String str = "       abc         ";
System.out.println("去掉左右空格后:" + str.trim());

4.9,字符串替换

? String replace(char oldChar,char newChar)
? String replaceAll(String,String)将某个内容全部替换成指定内容
? String repalceFirst(String,String)将第一次出现的某个内容替换成指定的内容

String str = "abcdefgabdc";
System.out.println("替换:" + str.replace("abc", "123"));
System.out.println("替换全部:" + str.replaceAll("ab", "12"));
System.out.println("替换第一次出现:" + str.repalceFirst("a", "a")); 

4.10,String substring(int beginIndex,int endIndex)

 String str = "abcdefg";
 // 截取0-3个位置的内容, 不含3
 System.out.println("截取后的字符为:" + str.substring(0, 3));
 // 从第3个位置开始截取, 含2
 System.out.println("截取后字符为:" + str.substring(2));

4.11,boolean equalsIgnoreCase(String)

String str = "ABC";
String str1 = "abc";
if(str.equalsIgnoreCase(str1)){
       System.out.println("相等");
}
else{
       System.out.println("不相等");
}

4.12,boolean contains(String)

String str = "ABCDEF";
String str1 = "ABC";
if(str.contains(str1)){
       System.out.println("str内容中包含ABC");
}
else{
       System.out.println("str内容中不包含ABC");
}

五,总结

? 1,对于JVM内存的分配,在jdk6中存在方法区,jdk8中便没有方法区,改成元区域。

? 2,jdk6中常量池存在方法区中,jdk7以后常量池移动到堆中。

? 本篇博客只是简单介绍了JVM内存分配和常量池的概念,仍有很多更深入的原理没有记录。因此在后期会继续更新关于JVM相关的原理知识,以上博客中如有不适之处还请留言(邮箱)指教。

感谢阅读!

?

?

原文地址:https://www.cnblogs.com/fenjyang/p/11449518.html

时间: 2024-10-12 04:33:48

JVM内存分配及String常用方法的相关文章

最简单例子图解JVM内存分配和回收

一.简介 JVM采用分代垃圾回收.在JVM的内存空间中把堆空间分为年老代和年轻代.将大量(据说是90%以上)创建了没多久就会消亡的对象存储在年轻代,而年老代中存放生命周期长久的实例对象.年轻代中又被分为Eden区(圣经中的伊甸园).和两个Survivor区.新的对象分配是首先放在Eden区,Survivor区作为Eden区和Old区的缓冲,在Survivor区的对象经历若干次收集仍然存活的,就会被转移到年老区. 简单讲,就是生命期短的对象放在一起,将少数生命期长的对象放在一起,分别采用不同的回收

【转】java内存分配和String类型的深度解析

一.引题 在java语言的所有数据类型中,String类型是比较特殊的一种类型,同时也是面试的时候经常被问到的一个知识点,本文结合java内存分配深度分析关于String的许多令人迷惑的问题.下面是本文将要涉及到的一些问题,如果读者对这些问题都了如指掌,则可忽略此文. 1.java内存具体指哪块内存?这块内存区域为什么要进行划分?是如何划分的?划分之后每块区域的作用是什么?如何设置各个区域的大小? 2.String类型在执行连接操作时,效率为什么会比StringBuffer或者StringBui

JVM内存分配简介

面向对象进阶 JVM内存分配 1.xxx.java----xxx.class (编译器进行) 2.xxx.class----类加载器(三种)----在内存分配空间 3.JVM会将内存分为5块① 栈:  Java栈区域比较小,空间大约1M. 特点:(存取数据)速度非常快,在栈里边执行的基本上都是快速的任务:基本数据类型和对象的引用.② 堆:  new出来的对象都存放在堆里边,堆里边的每一个对象都会产生一个唯一的地址. 内存地址会赋值给栈里边的引用.堆里边的对象需要垃圾回收机制才能销毁. 要销毁一个

深入理解JVM内存分配策略

理解JVM内存分配策略 三大原则+担保机制 JVM分配内存机制有三大原则和担保机制 具体如下所示: 优先分配到eden区 大对象,直接进入到老年代 长期存活的对象分配到老年代 空间分配担保 对象优先在Eden上分配 如何验证对象优先在Eden上分配呢,我们进行如下实验. 打印内存分配信息 首先代码如下所示: public class A { public static void main(String[] args) { byte[] b1 = new byte[4*1024*1024]; }

jvm内存模型,java类从编译到加载到执行的过程,jvm内存分配过程

一.jvm内存模型 JVM 内存模型主要分为堆.程序计数器.方法区.虚拟机栈和本地方法栈 1.堆 1.1.堆是 JVM 内存中最大的一块内存空间. 1.2.该内存被所有线程共享,几乎所有对象和数组都被分配到了堆内存中. 1.3.堆被划分为新生代和老年代,新生代又被进一步划分为 Eden 和 Survivor 区,最后 Survivor 由 From Survivor 和 To Survivor 组成. 2.程序计数器(Program Counter Register) 程序计数器是一块很小的内存

JVM虚拟机-03、JVM内存分配机制与垃圾回收算法

JVM虚拟机-03.JVM内存分配机制与垃圾回收算法 1 JVM内存分配与回收 1.1 对象优先在Eden区分配 大多数情况下,对象在新生代中?Eden?区分配.当?Eden?区没有足够空间进行分配时,虚拟机将发起一次Minor?GC.我们来进行实际测试一下.在测试之前我们先来看看?Minor?GC和Full?GC?有什么不同呢? Minor?GC/Young?GC:指发生新生代的的垃圾收集动作,MinorGC非常频繁,回收速度一般也比较快. Major?GC/Full?GC:一般会回收老年代,

Android性能调优篇之探索JVM内存分配

详细内容请查看我的简书地址:Android性能调优篇之探索JVM内存分配 或者我的个人博客地址:Android性能调优篇之探索JVM内存分配

JVM内存模型及String对象内存分配

昨天看了一篇关于<Java后端程序员1年工作经验总结>的文章,其中有一段关于String和StringBuffer的描述,对于执行结果仍然把握不准,趁此机会也总结了下JVM内存模型. 1.JVM运行时数据区域 关于JVM内存模型之前也了解过一些,也是看过就忘,好记性比如烂笔头,记下来吧.参考此文章http://chenzhou123520.iteye.com/blog/1585224 图1 JVM运行时数据区域 (1).程序计数器(Program Counter Register): 程序计数

jvm 内存分配 (转)

深入理解JVM-JVM内存模型 http://www.cnblogs.com/dingyingsi/p/3760447.html 我们知道,计算机CPU和内存的交互是最频繁的,内存是我们的高速缓存区,用户磁盘和CPU的交互,而CPU运转速度越来越快,磁盘远远跟不上CPU的读写速度,才设计了内存,用户缓冲用户IO等待导致CPU的等待成本,但是随着CPU的发展,内存的读写速度也远远跟不上CPU的读写速度,因此,为了解决这一纠纷,CPU厂商在每颗CPU上加入了高速缓存,用来缓解这种症状,因此,现在CP