先说说JDK API:

JDK中包含大量的API类库,所谓API(Application Programming Interface,应用程序编程接口,这些功能以类的形式封装)。

JDK API包含的类库功能强大,经常使用的有:字符串操作,集合操作,文件操作,输入输出操作,网络操作,多线程等等。

JDK包结构

围栏便于使用和维护,JDK类库按照包结构划分,不同功能的类划分在不同的包中;经常使用的包如下所示:

java.lang:Java程序的基础类,如字符串,多线程等,该包中的类使用是频率非常高,不需要import,可以直接使用
java.util:常用工具类,如集合,随机数产生器,日历,始终等等
java.io:文件操作,输入输出操作
java.net:网络操作
java.math:数学运算相关操作
java.security:安全相关操作
java.sql:数据库访问
java.text:处理文字,日期,数字,信息的格式
String基本概念
java.lang.String使用了final修饰,不能被继承;
字符串底层封装了字符数组及针对字符数组的操作算法;
字符串一旦创建,对象永远无法改变,但字符串引用可以重新赋值;
Java字符串在内存中采用Unicode编码方式,任何一个字符对应两个字节的定长编码;
String常量池:
Java为了提高性能,静态字符串(字面量/常量/常量连接的结果)在常量池中创建,并尽量使用同一个对象,重用静态字符串;
对于重复出现的字符串常量,JVM会首先在常量池中查找,如果存在即返回该对象。
String str1="Hello"; //创建对象//不会创建新的对象,而是使用常量池中已有的对象
String str2="Hello";
System.out.println(str1==str2); //输出true//使用new关键字会创建新的String对象
String str3=new String("Hello");
System.out.println(str1==str3);//输出false
内存编码及长度
String在内存中采用Unicode编码,每个字符占用两个字节;任何一个字符(无论中文还是英文)都算一个字符长度,占用两个字节。
使用indexOf实现检索:
indexOf方法用于实现在字符串中检索另一个字符串
String提供几个indexOf方法
int indexOf(String str):在字符串中检索str,返回其第一次出现的位置,如果找不到则返回-1
int indexOf(String str,int formIndex):从字符串formIndex位置开始检索
String还定义有lastIndexOf方法:
int lastIndexOf(String str,int form):str在字符串中多次出现时,将返回最后一个出现的位置
packagestringDemo;public classdemo1 {public static voidmain(String[] args) {
String str="I can because i think i can";int index=str.indexOf("can");
System.out.println(index);//2
index=str.lastIndexOf("can");
System.out.println(index);//24
index=str.lastIndexOf("can",6);
System.out.println(index);//24
index=str.indexOf("my"); //-1
}
}
使用substring获取字符串
substring方法用于返回一个字符串的子字符串
substring常量重载方法定义如下:
String substring(int beginIndex,int endIndex):返回字符串中从下标beginIndex(包括)开始到endIndex(不包括)结束的子字符串
String subString(int beginIndex):返回字符串中从下标beginIndex(包含)开始到字符串结尾的子字符串
packagestringDemo;public classdemo1 {public static voidmain(String[] args) {
String str="http://www.oracle.com";
String substr=str.substring(11,17);//oracle
System.out.println(substr);
substr=str.substring(7);
System.out.println(substr);//www.oracle.com
}
}
使用trim()方法去掉一个字符串的前导和后继空字符串
packagestringDemo;public classdemo1 {public static voidmain(String[] args) {
String userName=" good boy";
System.out.println(userName);
userName=userName.trim();
System.out.println(userName);
}
}
String中定义的charAt()方法:
char charAt(int index):方法charAt()用于返回字符串指定位置的字符,参数index表示指定的位置
packagestringDemo;public classdemo2 {public static voidmain(String[] args) {
String name="whatisjava";for(int i=0;i
System.out.print(c+"");//whatisjava
}
}
}
startsWith和endsWith方法:检测一个字符串是否以指定字符串开头或结尾
packagestringDemo;public classdemo2 {public static voidmain(String[] args) {
String name="whatisjava";
System.out.println(name.endsWith("java"));
System.out.println(name.startsWith("w"));
}
}
大小写变换toUpperCase和toLowerCase方法:转换字符串中英文字母的大小写形式:
packagestringDemo;public classdemo2 {public static voidmain(String[] args) {
String name="我喜欢Java";
name=name.toUpperCase();
System.out.println(name);
name=name.toLowerCase();
System.out.println(name);
}
}
valueOf方法:将其他方法转换为字符串类型
packagestringDemo;public classdemo2 {public static voidmain(String[] args) {double pi=3.1415926;int value=123;boolean flag=true;char[] charArr={'a','b','c'};
String str=String.valueOf(pi);
System.out.println(str);
str=String.valueOf(value);
System.out.println(str);
str=String.valueOf(flag);
System.out.println(str);
str=String.valueOf(charArr);
System.out.println(str);
}
}
分割字符串:
利用字符串类的split方法进行分割
packagestringDemo;public classdemo1 {public static voidmain(String[] args) {
String userName=" hello,good,boy";
System.out.println(userName);
String [] str=userName.split(",");for(String s:str)
System.out.println(s);
}
}
利用StringTokenizer来进行字符串分割
/** *//**利用StringTokenizer来进行字符串分割
*@paramstr 待分割的字符串
*@paramsdelimiter 分割符
*@return
*/
publicString[] useStringTokenizer(String str,String sdelimiter)...{
StringTokenizer token=newStringTokenizer(str,sdelimiter);
String[] array=newString[token.countTokens()];int i=0;while(token.hasMoreTokens())...{
array[i]=token.nextToken();
i++;
}returnarray;
}
字符串数组排序
/** *//**对字符串数组进行排序
*@paramstr 原始字符串数组
*@paramflag flag=0:顺序排序 flag=1:倒序排序
*@return排序后的字符串数组*/
public String[] sort(String[] str,intflag)...{if(str==null||str.length==0)throw newIllegalArgumentException();
String temp=str[0];//顺序排列 ,即从小到大
if(flag==0)...{for(int i=0;i0)...{
temp=str[i];
str[i]=str[j];
str[j]=temp;
}
}
}
}else if(flag==1)...{//倒序排列
for(int i=0;i
temp=str[i];
str[i]=str[j];
str[j]=temp;
}
}
}
}returnstr;
}
使用Hashtable对字符串进行碰撞
利用hashtable对字符串进行过滤,两个字符数组之间的比较,对字符串数组进行过滤
1.在一些字符串数组中,常会有重复的记录,比如手机号码,我们可以通过Hashtable来对其进行过滤
publicString[] checkArray(String[] str)...{
Hashtable hash=new Hashtable();for(int i=0;i
hash.put(str[i], str[i]);
}
Enumeration enumeration=hash.keys();
String[] str_new=newString[hash.size()];int i=0;while(enumeration.hasMoreElements())...{
str_new[i]=enumeration.nextElement().toString();
i++;
}returnstr_new;
}
2.A,B均为字符串数组,找出在A中存在,而在B中不存在的字符串
publicString[] compareArray(String[] A,String[] B){
Hashtable hash=newHashtable();
Hashtablehash_new=new Hashtable();for(int i=0;i
hash.put(B[i], B[i]);for(int i=0;i
hash_new.put(A[i], A[i]);
}
String[] C=newString[hash_new.size()];int i=0;
Enumeration enumeration=hash_new.keys();while(enumeration.hasMoreElements()){
C[i]=enumeration.nextElement().toString();
i++;
}returnC;
}
3.将一个字符串数组中某一个特定的字符串过滤掉
/** *//**检验一个字符串数组,若包含某一特定的字符串,则将该字符串从数组中删
除,返回剩余的字符串数组
*@paramstr_array 字符串数组
*@paramstr_remove 待删除的字符串
*@return过滤后的字符串*/
publicString[] removeStrFromArray(String[] str_array,String
str_remove)...{
Hashtable hash=new Hashtable();for(int i=0;i
hash.put(str_array[i], str_array[i]);
}//生成一个新的数组
String[] str_new=newString[hash.size()];int i=0;
Enumeration enumeration=hash.keys();while(enumeration.hasMoreElements())...{
str_new[i]=enumeration.nextElement().toString();
i++;
}returnstr_new;
}
StringBuild
StringBuilder封装可变字符串,对象创建后可以通过调用方法改变其封装的字符序列。
StringBuilder有如下常用构造方法:
public StringBuilder()
public StringBuilder(String str )
StringBuilder常用方法:
StringBuilder append(String str):追加字符串.
packagestringDemo;public classdemo3 {public static voidmain(String[] args) {
StringBuilder stb=new StringBuilder("hello");
System.out.println(stb);
stb=stb.append("word");
System.out.println(stb);
}
}
StringBuilder insert(int desOffset,String s):插入字符串
packagestringDemo;public classdemo3 {public static voidmain(String[] args) {
StringBuilder stb=new StringBuilder("hello");
System.out.println(stb);
stb=stb.insert(3, "word");
System.out.println(stb);
}
}
StringBuilder delete(int start,int end):删除字符串
packagestringDemo;public classdemo3 {public static voidmain(String[] args) {
StringBuilder stb=new StringBuilder("hello");
stb=stb.delete(2, 5);
System.out.println(stb);
}
}
StringBuilder replace(int start,int end,String s):替换字符串
packagestringDemo;public classdemo3 {public static voidmain(String[] args) {
StringBuilder stb=new StringBuilder("hello");
stb=stb.delete(2, 5);
System.out.println(stb);
}
}
StringBuilder reverse():字符串反转
packagestringDemo;public classdemo3 {public static voidmain(String[] args) {
StringBuilder stb=new StringBuilder("hello");
System.out.println(stb);//hello
stb=stb.reverse();
System.out.println(stb);//olleh
}
}
注意:
StringBuilder的很多方法的返回值均为StringBuilder类型,这些方法的返回语句为:return this.
由于改变封装的字符序列后又返回了该对象的引用,因此可以按照简洁的方式书写代码:
buf.append("ibm").insert(3,"oracle").repalce(9,13,"java");
StringBuilder是可变字符串,字符串的内容计算建议使用StringBuillder实现,这样性能会好一些。
Java的字符串连接过程是利用StringBuilder实现的。
StringBuffer常用方法
(StringBuffer和StringBuilder在使用上几乎一样)
StringBuffer s = new StringBuffer();
这样初始化出的StringBuffer对象是一个空的对象,
StringBuffer sb1=new StringBuffer(512);
分配了长度512字节的字符缓冲区。
StringBuffer sb2=new StringBuffer(“how are you?”)
创建带有内容的StringBuffer对象,在字符缓冲区中存放字符串“how are you?”
a、append方法
public StringBuffer append(boolean b)
该方法的作用是追加内容到当前StringBuffer对象的末尾,类似于字符串的连接,调用该方法以后,StringBuffer对象的内容也发生改 变,例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.append(true);
则对象sb的值将变成”abctrue”
使用该方法进行字符串的连接,将比String更加节约内容,经常应用于数据库SQL语句的连接。
b、deleteCharAt方法
public StringBuffer deleteCharAt(int index)
该方法的作用是删除指定位置的字符,然后将剩余的内容形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“KMing”);
sb. deleteCharAt(1);
该代码的作用删除字符串对象sb中索引值为1的字符,也就是删除第二个字符,剩余的内容组成一个新的字符串。所以对象sb的值变 为”King”。
还存在一个功能类似的delete方法:
public StringBuffer delete(int start,int end)
该方法的作用是删除指定区间以内的所有字符,包含start,不包含end索引值的区间。例如:
StringBuffer sb = new StringBuffer(“TestString”);
sb. delete (1,4);
该代码的作用是删除索引值1(包括)到索引值4(不包括)之间的所有字符,剩余的字符形成新的字符串。则对象sb的值是”TString”。
c、insert方法
public StringBuffer insert(int offset, boolean b),
该方法的作用是在StringBuffer对象中插入内容,然后形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“TestString”);
sb.insert(4,false);
该示例代码的作用是在对象sb的索引值4的位置插入false值,形成新的字符串,则执行以后对象sb的值是”TestfalseString”。
d、reverse方法
public StringBuffer reverse()
该方法的作用是将StringBuffer对象中的内容反转,然后形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.reverse();
经过反转以后,对象sb中的内容将变为”cba”。
e、setCharAt方法
public void setCharAt(int index, char ch)该方法的作用是修改对象中索引值为index位置的字符为新的字符ch。例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.setCharAt(1,’D’);
则对象sb的值将变成”aDc”。
f、trimToSize方法
public void trimToSize()
该方法的作用是将StringBuffer对象的中存储空间缩小到和字符串长度一样的长度,减少空间的浪费,和String的trim()是一样的作用,不在举例。
g、length方法
该方法的作用是获取字符串长度 ,不用再说了吧。
h、setlength方法
该方法的作用是设置字符串缓冲区大小。
StringBuffer sb=new StringBuffer();
sb.setlength(100);
如果用小于当前字符串长度的值调用setlength()方法,则新长度后面的字符将丢失。
i、sb.capacity方法
该方法的作用是获取字符串的容量。
StringBuffer sb=new StringBuffer(“string”);
int i=sb.capacity();
j、ensureCapacity方法
该方法的作用是重新设置字符串容量的大小。
StringBuffer sb=new StringBuffer();
sb.ensureCapacity(32); //预先设置sb的容量为32
k、getChars方法
该方法的作用是将字符串的子字符串复制给数组。
getChars(int start,int end,char chars[],int charStart);
StringBuffer sb = new StringBuffer("I love You");
int begin = 0;
int end = 5;
//注意ch字符数组的长度一定要大于等于begin到end之间字符的长度
//小于的话会报ArrayIndexOutOfBoundsException
//如果大于的话,大于的字符会以空格补齐
char[] ch  = new char[end-begin];
sb.getChars(begin, end, ch, 0);
System.out.println(ch);

结果:I lov

String、StringBuffer、StringBuilder区别

StringBuffer、StringBuilder和String一样,也用来代表字符串。String类是不可变类,任何对String的改变都 会引发新的String对象的生成;StringBuffer则是可变类,任何对它所指代的字符串的改变都不会产生新的对象。既然可变和不可变都有了,为何还有一个StringBuilder呢?相信初期的你,在进行append时,一般都会选择StringBuffer吧!

先说一下集合的故事,HashTable是线程安全的,很多方法都是synchronized方法,而HashMap不是线程安全的,但其在单线程程序中的性能比HashTable要高。StringBuffer和StringBuilder类的区别也是如此,他们的原理和操作基本相同,区别在于StringBufferd支持并发操作,线性安全的,适 合多线程中使用。StringBuilder不支持并发操作,线性不安全的,不适合多线程中使用。新引入的StringBuilder类不是线程安全的,但其在单线程中的性能比StringBuffer高。

注意:不必考虑到线程同步问题,我们应该优先使用StringBuilder类;如果要保证线程安全,自然是StringBuffer。

效率:StringBuilder>StringBuffer>String

相关面试题:

1)String是最基本的数据类型吗?

基本数据类型包括byte,int, char ,long ,float ,double ,boolean和short.

String是引用数据类型

java.lang.String类是final类型的,因此不可以继承这个类,不能修改这个类,为了提高效率节省空间,我们应该用StringBuffer / StringBuilder类

2)String和StringBuffer的区别

这两个类都实现了CharSequence接口。

(1)类型不同,因为不是一个类,也没有继承关系,做参数时不能共用

(2)String对象是不可变对象,不能修改值。而StringBuffer是可变对象,能修改值。

(3)拼接字符串时,String会产生新对象,而StringBuffer只是增加新字符,不产生新对象,因此效率高。

(4)String覆盖了equals方法和hashCode方法,而StringBuffer没有覆盖equals方法和hashCode方法,所以,将StringBuffer对象存储进Java集合类中会出现问题。