创建String字符串
String greeting = "JunSouth";
前言
Java虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。JVM主要管理堆和非堆两种类型内存
简单来说,非堆包含方法区、JVM内部处理或优化所需的内存(编译后的代码缓存)、每个类结构(如运行时常数池、字段和方法数据)以及方法和构造方法的代码。Java的堆是一个运行时数据区,类的对象从中分配空间。这些对象通过new、newarray等指令建立。因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。由于要在运行时动态分配内存,存取速度较慢。
栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量数据(int, short, long, byte, float, double, boolean, char)
虚拟机必须为每个被装载的类型维护一个常量池。常量池就是该类型所用到常量的一个有序集合,包括直接常量(string,integer和 floating point常量)和对其他类型,字段和方法的符号引用。
对于String常量,它的值是在常量池中的。而JVM中的常量池在内存当中是以表的形式存在的,
对于String类型,有一张固定长度的CONSTANT_String_info表用来存储文字字符串值,该表只存储文字字符串值,不存储符号引用。在程序执行的时候,常量池会储存在Method Area,而不是堆中。常量池中保存着很多String对象; 并且可以被共享使用,因此它提高了效率。
常用String的方法
length() | 求字符串的长度 |
---|---|
indexOf() | 求某个字符在字符串中的位置 |
charAt() | 求一个字符串中某个位置的值 |
equals() | 比较两个字符串是否相同 |
replace() | 将字符串中的某些字符用别的字符替换掉。形如replace(“abc”,”ddd”);字符串中的abc将会被ddd替换掉 |
split() | 根据给定正则表达式的匹配拆分此字符串。形如 String s = "The time is going quickly!"; str1=s.split(" ") |
substring(int,int) | 输出一个新的字符串,它是此字符串中的子串,形如substring(3,7)把字符串中的第四至六位输出 |
substring(int) | 从指定下标开始一直截取到字符串的最后 |
trim() | 将字符串开头的空白(空格)和尾部的空白去掉 |
format() | 使用指定的语言环境、格式字符串和参数返回一个格式化字符串 |
toLowerCase() | 将字符串中所有的大写改变成小写 |
toUpperCase() | 将字符串中所有的小写改变为大写 |
concat() | 连接两个字符串 |
indexOf() | 得到指定内容第一次出现的下标 |
lastIndexOf() | 得到指定内容最后一次出现的下标 |
valueOf() | 转换为字符串 |
tartsWith(String) endsWith(String) |
判断一个字符串是否以指定的内容开头 boolean 判断一个字符串是否以指定的内容结尾 |
public static void main(String[] args) {
String str1 = new String("abcdef");//初始化一个String对象
System.out.println(str1.length());//输出字符串的长度
System.out.println(str1.indexOf("c"));//输出指定字符在字符串中的位置
System.out.println(str1.charAt(3));//输出指定位置的值
String str2 = new String("JunSouth");
System.out.println(str1.equals(str2));//比较两个字符串是否是相等的。
String str3 = new String("abcdefghijk");
System.out.println(str3);//输出str3的字符串
System.out.println( str3.replace("def","fed"));//输出替换之后的字符串
}
public static void main(String[] args) {
//split()
String[] str1;//声明一个空的数组
String s = "The time is going quickly!";//声明一个字符串
str1=s.split(" ");//根据给定正则表达式的匹配拆分字符串。
for (int i=0;i<str1.length;i++){
System.out.println(str1[i]);
}
str1 = s.split(" ",3);//根据匹配给定的正则表达式来拆分字符串。
for (int i=0;i<str1.length;i++){
System.out.println(str1[i]);
}
//substring()
System.out.println(s.substring(4,7));//预计输出tim
//trim()
String ss = " dssd fkdj ";
System.out.println(ss.trim());//输出字符串前面的空白处
//format
int d = 2018;
String sd = "Study Hard!";
System.out.println(String.format("%d年加油,%s",d,sd));//将其他类型的数据和字符串按照指定的格式输出
//toLowerCase
//toUpperCase
System.out.println(sd.toLowerCase());//将所有大写字母变为小写
System.out.println(sd.toUpperCase());//将所有小写字母变为大写
}
public static void main(String[] args) {
/**
* 情景一:字符串池
* JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象;
* 并且可以被共享使用,因此它提高了效率。
* 由于String类是final的,它的值一经创建就不可改变。
* 字符串池由String类维护,我们可以调用intern()方法来访问字符串池。
*/
//在字符串池创建了一个对象
String s1 = "abc";
//字符串pool已经存在对象“abc”(共享),所以创建0个对象,累计创建一个对象
String s2 = "abc";
//true 指向同一个对象,
System.out.println("s1 == s2 : "+(s1==s2));
//true 值相等
System.out.println("s1.equals(s2) : " + (s1.equals(s2)));
/**
* 情景二:关于new String("")
*/
//创建了两个对象,一个存放在字符串池中,一个存在与堆区中;
//还有一个对象引用s3存放在栈中
String s3 = new String("abc");
//字符串池中已经存在“abc”对象,所以只在堆中创建了一个对象
String s4 = new String("abc");
//false s3和s4栈区的地址不同,指向堆区的不同地址;
System.out.println("s3 == s4 : "+(s3==s4));
//true s3和s4的值相同
System.out.println("s3.equals(s4) : "+(s3.equals(s4)));
//false 存放的地区多不同,一个栈区,一个堆区
System.out.println("s1 == s3 : "+(s1==s3));
//true 值相同
System.out.println("s1.equals(s3) : "+(s1.equals(s3)));
/**
* 情景三
* 由于常量的值在编译的时候就被确定(优化)了。
* 在这里,"ab"和"cd"都是常量,因此变量str3的值在编译时就可以确定。
* 这行代码编译后的效果等同于: String str3 = "abcd";
*/
String str1 = "ab" + "cd"; //1个对象
String str11 = "abcd";
System.out.println("str1 = str11 : "+ (str1 == str11));
/**
* 情景四:
* 局部变量str2,str3存储的是存储两个拘留字符串对象(intern字符串对象)的地址。
*
* 第三行代码原理(str2+str3):
* 运行期JVM首先会在堆中创建一个StringBuilder类,
* 同时用str2指向的拘留字符串对象完成初始化,
* 然后调用append方法完成对str3所指向的拘留字符串的合并,
* 接着调用StringBuilder的toString()方法在堆中创建一个String对象,
* 最后将刚生成的String对象的堆地址存放在局部变量str3中。
*
* 而str5存储的是字符串池中"abcd"所对应的拘留字符串对象的地址。
* str4与str5地址当然不一样了。
*
* 内存中实际上有五个字符串对象:
* 三个拘留字符串对象、一个String对象和一个StringBuilder对象。
*/
String str2 = "ab"; //1个对象
String str3 = "cd"; //1个对象
String str4 = str2+str3;
String str5 = "abcd";
// false
System.out.println("str4 = str5 : " + (str4==str5));
/**
* 情景五:
* JAVA编译器对string + 基本类型/常量 是当成常量表达式直接求值来优化的。
* 运行期的两个string相加,会产生新的对象的,存储在堆(heap)中
*/
String str6 = "b";
String str7 = "a" + str6;
String str67 = "ab";
//str6为变量,在运行期才会被解析。
System.out.println("str7 = str67 : "+ (str7 == str67));
final String str8 = "b";
String str9 = "a" + str8;
String str89 = "ab";
//str8为常量变量,编译期会被优化
System.out.println("str9 = str89 : "+ (str9 == str89));
}