String 类及其创建
String 类的创建
String 类是 Java 内置的一个类,其完全限定类名是java.lang.String
。想要创建一个字符串有多重方式,比如创建字符串"Hello"
:
String s1 = "Hello"; // 字面量创建
String s2 = new String("Hello"); // 构造函数创建
String s3 = "He" + "llo"; // 字符串连接
String s4 = new String(new char[]{'H','e','l','l','o'}); // 字符数组创建
String s5 = s4.intern();
String 对象在内存中的位置
上面的s1,s3
存在常量池里,而s2,s4
存在堆里。类似于 C 程序的编译,可能编译器在编译String s1 = "Hello"
时,将"Hello"
存进了可重定位文件的.rodata
(只读数据节)中了,并且将这个字符串与s1
的联系写进重定位节。s3
在编译时预处理,自动计算了"He" + "llo"
并也存入.rodata
;由于已经有一个"Hello"
了,就不再重复加入"Hello"
,而是直接将s3
与之前的那个"Hello"
进行重定位。
s2,s4
都是通过new
一个新的字符串对象得到的,这个就好比C
语言的malloc
,分配的空间在内存的堆中。
s5
也是存在常量池里,和s1,s3
一样。intern
是String
类的一个实例方法,返回字符串对象的字面量。下面是这个方法的 API 注释:
从框起来的部分可以看到,intern
方法对于str2=str1.intern()
处理在内存分配上有两种情况:
1 1 1.str1
的字面量已经在常量池里,则直接让str2
指向它。此时有str1==str2
。
2 2 2.str1
的字面量不在常量池里,这时str1
肯定是在堆里面分配的,JVM会把str1
的字面量存入常量池,再令str2
指向它。
所以不管怎么说,通过intern
方法得到的引用对象,始终指向常量池。
String 类的一些常用方法
字符串比较
equals
。s1.equals(s2)
比较s1
与s2
的字面量,如果字面量相等就返回true
,否则是false
。如果s2==null
也是返回false
。
在 Java 中由于所有非基本数据类型的变量都是引用变量,
s1==s2
这一语句其实是在判断s1
和s2
是否指向同一个内存区。相当于 C 语言中两个指针的==
操作。
equalsIgnoreCase
。忽略大小写的比较。regionMatches
。比较部分内容是否相同,这个函数有两个重载的写法:regionMatches(int toffset,String other,int ooffset,int len)
。调用s1
的该方法,比较s1[toffset]
和other[ooffset]
开始的len
个字符是否都相等;如果都相等返回true
,否则false
。
String s0 = "Java"; String s2 = "Welcome to Java"; String s3 = "welcome to java"; System.out.println(s2.regionMatches(11, s0, 0, 4) ); //true System.out.println(s3.regionMatches(11, s0, 0, 4)); //false
regionMatches(boolean ignoreCase, int toffset, String other, int ooffset, int len)
。和上面那个方法用法类似,只是当ignoreCase==true
时比较的时候忽略大小写。
System.out.println(s3.regionMatches(true, 11, s0, 0, 4));//true
startsWith(s1)
判断是否以字符串s1
开始。endsWith(s1)
判断是否以字符串s1
结束。compareTo
方法用于比较两个字符串的大小,即第一个不同字符的差值(字典序)。
字符串的长度
使用length()
方法可以返回字符串的长度。注意数组获取长度是length
成员,两者不一样。
获取特定位置的字符
如果是 C 语言直接就是s[index]
了,但是 Java 要用s.charAt(index)
。
连接字符串
可以直接用加法连接:
String s3 = s1 + s2;
也可以是调用函数concat
String s3 = s1.concat(s2);
两者效果一样,但是不会改变s1,s2
的值。连接操作返回的是一个独立于s1,s2
的新的字符串。
截取字符串
调用subString
方法。有两个重载的subString
方法:
substring(int beginIndex, int endIndex)
返回从beginIndex
到endIndex-1
的片段。substring(int beginIndex)
返回从beginIndex
到末尾的片段。
字符串转换
s1.toLowerCase()
将s1
转换成小写形式,得到新串。s1.toUpperCase()
将s1
转换成大写形式,得到新串。s1.trim()
删除s1
两端的空格,得到新串。s1.replace(oldChars,newChars)
用串newChars
替换s1
中的所有子串oldChaes
,得到新串。
查找字符和字符串
indexOf
返回字符串中字符或字符串匹配的位置,返回-1
表示未找到。
"Welcome to Java".indexOf('W') returns 0.
"Welcome to Java".indexOf('x') returns -1.
"Welcome to Java".indexOf("come") returns 3.
indexOf
还能加一个参数fromIndex
,或者加两个参数beginIndex,endIndex
,表示在特定的区间寻找字符/字符串。
"Welcome to Java".indexOf('o‘,5) returns 9.
"Welcome to Java".indexOf("Java", 5) returns 11.
"Welcome to Java".indexOf("java", 5) returns -1.
lastIndexOf
从字符串末尾开始查找。
"Welcome to Java".lastIndexOf('a') returns 14.
字符数组与字符串的转换
- 字符串到字符数组,用
toCharArray
。
String s = "Java";
char[ ] charArray = s.toCharArray( );// charArray.length=4
- 字符数组到字符串,用构造函数,或者静态方法
valueOf
:
String s = new String(new char[ ] {'J','a','v','a'} ); // "Java"
String s = String.valueOf( new char[ ] {'J','a','v','a'} ); // "Java"
基本数据类型和字符串间的转换
valueOf
方法将基本数据类型转换为字符串
String s1 = String.valueOf(1.0); //"1.0"
String s2 = String.valueOf(true); //"true"
- 字符串转换为基本类型:利用包装类
double d = Double.parseDouble(str);
int i = Integer.parseInt(str);
boolean b = Boolean.parseBoolean(str);