自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

JVM內(nèi)存分配及String常用方法

云計算 虛擬化
在介紹String類之前,先來簡單分析一下在JVM中,對內(nèi)存的使用是如何進行分配的。JVM將內(nèi)存分為多個不同的區(qū)域,這些區(qū)域都有各自的用途、創(chuàng)建和銷毀的時間,有些區(qū)域隨虛擬機進程的啟動而存在,有些區(qū)域則是依賴用戶線程的啟動和結(jié)束來建立和銷毀。

[[275905]]

 一,JVM內(nèi)存分配和常量池

​ 在介紹String類之前,先來簡單分析一下在JVM中,對內(nèi)存的使用是如何進行分配的。如下圖所示(注意:在jdk1.8之后便沒有方法區(qū)了):

 

​ 如上JVM將內(nèi)存分為多個不同的區(qū)域,這些區(qū)域都有各自的用途、創(chuàng)建和銷毀的時間,有些區(qū)域隨虛擬機進程的啟動而存在,有些區(qū)域則是依賴用戶線程的啟動和結(jié)束來建立和銷毀。

區(qū)域名稱的說明:

1.1,方法區(qū):

​ 屬于數(shù)據(jù)共享內(nèi)存區(qū)域,存儲已被虛擬機加載的類信息、常量、靜態(tài)變量、即時編譯器編譯后的代碼等數(shù)據(jù)。

1.2,虛擬機棧

​ 虛擬機棧就是我們通常說的棧,是Java執(zhí)行方法的內(nèi)存模型,每當執(zhí)行一次方法時,都會創(chuàng)建一個棧幀。把棧幀壓入棧,當Java方法調(diào)用時返回正常的結(jié)果或者捕獲異常時,棧幀出棧。

​ 棧幀:棧幀存儲方法的相關(guān)信息,包含局部變量數(shù)表、返回值、操作數(shù)棧、動態(tài)鏈接。

1.3,本地方法棧

​ 從功能上來說與虛擬機棧類似,但是虛擬機棧執(zhí)行的是字節(jié)碼,而本地方法棧調(diào)用的是Native方法,并且它是線程獨享的。

1.4,程序計數(shù)器

​ 程序計數(shù)器是線程獨享的,它是記錄當前線程執(zhí)行的字節(jié)碼行號。在多線程執(zhí)行時,CPU會來回在線程之間進行切換,那么當再次回到一條線程時,是如何得知線程的存儲單元及執(zhí)行指令。而程序計數(shù)器便會進行存儲下一條存儲單元的地址,執(zhí)行完畢后程序計數(shù)器自動加 1 ,以此循環(huán)直到程序結(jié)束為止。

1.5,堆

​ 說到堆這個概念想必都不陌生,它是內(nèi)存中的重要角色。它主要是用來存儲被創(chuàng)建出來的對象,通過關(guān)鍵字new實例出來的,是所有線程共享的一塊最大的區(qū)域。

​ ==特別注意:在JDK1.7及以后,常量池移動到堆內(nèi)存中。==

​ 堆還包括一個==常量池==,用來存儲編譯期間生成的==字面量和符號==引用。這部分內(nèi)容在類被加載后,都會存儲到方法區(qū)中。同時,運行時產(chǎn)生的新常量也可以被放入常量池中,比如 String 類中的 intern() 方法產(chǎn)生的常量。

​ 常量池就是這個類型用到的常量的一個有序集合。包括直接常量(基本類型,String)和對其他類型、方法、字段的符號引用。

二,常量池

2.1,什么是常量:

​ 常量是指被final修飾的變量,值一旦確定就無法改變。

​ final可以修飾靜態(tài)變量、方法、實例變量和局部變量。

​ 常量池分為兩種形式:靜態(tài)常量池和運行時常量池

2.2,靜態(tài)常量池

​ 即*.class文件中的常量池,class文件中的常量池不僅僅包含字符串(數(shù)字)字面量,還包含類、方法的信息,占用class文件絕大部分空間。這種常量池用于存放字面量和符號引用量。

2.3,運行時常量池

​ 指JVM虛擬機在完成類裝載操作后,將class文件中的常量池載入到內(nèi)存中,并保存在方法區(qū)中,我們常說的常量池,就是指方法區(qū)中的運行時常量池。同樣運行時常量池一個重要的特征就是具有動態(tài)性,指并不需要常量只有在編譯期才會產(chǎn)生,在運行期也會將新的常量保存到常量池中,如String類中的intern()方法。

三,== 和equals

3.1,兩者之間區(qū)邊

​ ==:

​ 對于基本類型來說:==表示數(shù)值的比較

​ 對于引用類型來說:==表示地址值的比較

​ equals:

​ 比較的是兩者之間值是否相等,但是Java中的類都是直接或者間接繼承Object類,而equals不也例外。其實在equals源碼中也是使用==進行比較的,如下源碼:

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

​ 那么問題來了,這和==又有什么區(qū)別呢?

​ 上面說到equals也是繼承自java.lang.Object,因此可以對equals進行重寫來定義我們自己的比較方式。

​ 請參看以下代碼:

  1. String str1 = "abc"
  2.         String str2 = "abc"
  3.  
  4.         char[] strArray = {'a','b','c'}; 
  5.         String str3 = new String(strArray); 
  6.  
  7.         String str4 = "abc"
  8.  
  9.         System.out.println(str1 == str2);    
  10.         System.out.println(str1 == str3);    
  11.         System.out.println(str2 == str3);    
  12.         System.out.println(str4.equals(str1));  

​ 以上運行結(jié)果為:

  1. true 
  2. false 
  3. false 
  4. true 

​ 接下來我們依次分析上面的結(jié)果:

​ 1,str1與str2比較的是字符串對象地址,因為它們的值是相同的,所以地址值也是相同的。

​ 2,str3是new出來的示例對象,在堆內(nèi)存中會開辟一塊新的內(nèi)存地址,它并不在常量池中。所以返回結(jié)果為false。

​ 3,同理str2與str3比較也是一樣的結(jié)果。

​ 4,equals比較的是值是否相同,所以返回的結(jié)果為true。

如圖所示:

 

四,String常用方法

​ 首先聲明字符串:

  1. String str1 = "abc"

4.1,int length()

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

4.2,char charAt(值)

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

4.3,char toCharArray()

  1. String str= "abc"; 
  2. char c[] = str.toCharArray();   
  3. for (int i = 0; i < c.length; i++) { 
  4.       System.out.println("轉(zhuǎn)為數(shù)組輸出:" + c[i]); 

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

  1. String str="axcdefgabc"
  2. int a1 = str.indexOf("a");  
  3. int a2 = str.indexOf("x",  2); 
  4. int a3 = str.lastIndexOf("c");  
  5. System.out.println("你的位置為:" + a1);  
  6. System.out.println("為的位置為:" + a2); 
  7. System.out.println("點最后出現(xiàn)的位置為:" + a3); 

4.5,字符串大小寫轉(zhuǎn)換

​ toUpperCase(); 轉(zhuǎn)換成大寫

​ toLowerCase();轉(zhuǎn)換成小寫

  1. String str = "hello world"
  2. String str1 = "HELLO WORD"
  3. System.out.println("將字符串轉(zhuǎn)大寫為:" + str.toUpperCase()); 
  4. System.out.println("將字符串轉(zhuǎn)換成小寫為:" + str1.toLowerCase()); 

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

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

4.7,boolean equals(Object anObject)

  1. String str = "abc"
  2.  String str1= "123";   
  3.  if(str.equals(str1)) {   
  4.       System.out.println("相等");  
  5.  }   
  6.  else
  7.       System.out.println("不相等");   
  8.  } 

4.8,String trim()

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

4.9,字符串替換

  1. String replace(char oldChar,char newChar) 
  2. ​ String replaceAll(String,String)將某個內(nèi)容全部替換成指定內(nèi)容 
  3. ​ String repalceFirst(String,String)將第一次出現(xiàn)的某個內(nèi)容替換成指定的內(nèi)容 
  4.  
  5. String str = "abcdefgabdc";   
  6. System.out.println("替換:" + str.replace("abc""123"));  
  7. System.out.println("替換全部:" + str.replaceAll("ab""12"));  
  8. System.out.println("替換第一次出現(xiàn):" + str.repalceFirst("a""a"));  

4.10,String substring(int beginIndex,int endIndex)

  1. String str = "abcdefg";   
  2.  // 截取0-3個位置的內(nèi)容, 不含3  
  3.  System.out.println("截取后的字符為:" + str.substring(0, 3)); 
  4.  // 從第3個位置開始截取, 含2 
  5.  System.out.println("截取后字符為:" + str.substring(2)); 

4.11,boolean equalsIgnoreCase(String)

  1. String str = "ABC"
  2. String str1 = "abc"
  3. if(str.equalsIgnoreCase(str1)){ 
  4.        System.out.println("相等"); 
  5. else
  6.        System.out.println("不相等"); 

4.12,boolean contains(String)

  1. String str = "ABCDEF"
  2. String str1 = "ABC"
  3. if(str.contains(str1)){ 
  4.        System.out.println("str內(nèi)容中包含ABC"); 
  5. else
  6.        System.out.println("str內(nèi)容中不包含ABC"); 

五,總結(jié)

​ 1,對于JVM內(nèi)存的分配,在jdk6中存在方法區(qū),jdk8中便沒有方法區(qū),改成元區(qū)域。

​ 2,jdk6中常量池存在方法區(qū)中,jdk7以后常量池移動到堆中。

責任編輯:武曉燕 來源: 博客園
相關(guān)推薦

2012-01-11 10:45:57

JavaJVM

2010-09-25 15:40:52

配置JVM內(nèi)存

2018-04-08 08:45:53

對象內(nèi)存策略

2009-07-09 10:01:26

設置JVM內(nèi)存分配

2022-01-07 13:50:55

語言內(nèi)存代碼

2009-06-12 09:46:40

Java String

2020-05-27 21:13:27

JavaJVM內(nèi)存

2010-10-19 14:48:09

Java String

2011-12-20 10:43:21

Java

2021-10-15 08:51:09

Linux內(nèi)存 Kmalloc

2010-09-27 13:41:22

JVM內(nèi)存回收

2023-03-26 00:43:42

JVM對象測試

2018-12-20 10:17:35

JVM模型內(nèi)存溢出

2010-09-26 15:38:33

JVM內(nèi)存泄漏

2021-07-30 07:22:51

JVM虛擬機棧 Stack

2010-09-27 11:00:27

TomcatJVM內(nèi)存

2010-02-22 08:58:35

JVM內(nèi)存模型垃圾收集

2009-08-27 18:04:01

c#擴展方法string

2019-09-02 14:53:53

JVM內(nèi)存布局GC

2025-03-03 09:05:56

點贊
收藏

51CTO技術(shù)棧公眾號