Java基础教程——String类

String类

Java程序中的所有字符串字面值(如 "abc" )都是String的实例
字符串是常量(因为 String 对象是不可变的,所以可以共享)
字符串的本质是字符数组:private final char value[];

创建字符串常用的方式

public class CreateString {
	public static void main(String[] args) {
		// 直接定义字符串
		String str1 = "虎老狮";
		// new 字符串
		String str2 = new String("Java");
		// 字符数组→字符串
		char[] arrChar = { 'A', 'n', 'd', 'y' };
		String str3 = new String(arrChar);
		// 字节数组→字符串
		byte[] arrByte = { 97, 98 };
		String str4 = new String(arrByte);
		System.out.println(str1);
		System.out.println(str2);
		System.out.println(str3);
		System.out.println(str4);
	}
}

字符串的比较

public class 字符串比较 {
	public static void main(String[] args) {
		String s1 = "奎木狼";
		String s2 = "奎木狼";
		String s3 = new String("奎木狼");
		System.out.println(s1 == s2);
		System.out.println(s1 == s3);
	}
}

解析:

采用字面值创建的字符串存放在常量池中(constant pool),同样内容的字符串对象指向同一个字符串对象。
常量池中的字符串,无则创建,有则直接使用。

字符串常量池:
相当于缓存,可以使得程序运行更快
以前在方法区,JDK 1.7之后,移到堆内存区(暂时不理解Java虚拟机内存知识的话,这句话直接无视)

new String()在运行时创建字符串对象,不在常量池中,因此s3和s1、s2不是同一个对象。

关于使用==进行比较判断:

|--基本类型,数值比较(只要求值相等,类型可以不同)
|--引用类型,地址比较(只有指向同一个对象才返回true)

public class 双等号基本类型 {
	public static void main(String[] args) {
		int a = 97;
		double b = 97.0;
		char c = 'a';
		System.out.println(a == b);// true
		System.out.println(c == b);// true
	}
}
equals方法

String类提供equals方法进行字符串比较。
看下这个方法的源码:

地址相等的时候,直接就判为相等。地址不相等时,长度不等的直接判为不相等;长度相等时,一个一个字符比较。

    public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof String) {
            String anotherString = (String)anObject;
            int n = value.length;
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

应用:

public class 字符串比较 {
	public static void main(String[] args) {
		String s1 = "奎木狼";
		String s2 = "奎木狼";
		String s3 = new String("奎木狼");
		System.out.println(s1.equals(s3));
		System.out.println(s3.equals("奎木狼"));// 不推荐,可能空指针异常
		System.out.println("奎木狼".equals(s3));// 推荐
	}
}

推荐使用字符串字面值调用equals方法,即使用"...".equals(...),因为引号中的字符串字面值一定不会为null,调用方法不会出现空指针异常。
但是很多时候,无法确定其是否为null(比如调用equals方法的对象是从别处传入的参数)。
如果不能确定调用equals的对象必不为null,就需要在调用之前先做null判断。

if (s3 != null) {
	boolean equals = s3.equals(s2);
	System.out.println(equals);
}

或者使用Objects类提供的equals方法进行比较。Objects类从JAVA 1.7开始提供,可以防止空指针异常。

		String s1 = "奎木狼";
		String s2 = "奎木狼";
		String s3 = new String("奎木狼");
		boolean equals = Objects.equals(s2, s3);
		System.out.println(equals);// true

更多字符串比较方法:
boolean equalsIgnoreCase(String anotherString) 判断字符串anotherString是否与当前字符串相等,忽略大小写形式
int compareTo(String anotherString) 根据ASCII码比较字符串anoterString和当前字符串的大小,比较方式类似于C语言中的strcmp函数
boolean startsWith(String prefix) 判断当前字符串是否以字符串prefix为开头
boolean endsWith(String suffix) 判断当前字符串是否以字符串suffix为后缀
public class Test字符串比较 {
	public static void main(String[] args) {
		String str1 = "ABC";
		System.out.println(str1.equalsIgnoreCase("abc"));// equalsIgnoreCase比较重要
		System.out.println(str1.compareTo("ABB"));// 1
		System.out.println(str1.compareTo("ABC"));// 0
		System.out.println(str1.compareTo("ABD"));// -1
		System.out.println(str1.startsWith("AB"));// T
		System.out.println(str1.endsWith("C"));// T
	}
}

字符串变量重新赋值,其实原字符串内容不变,只是变量中引用的地址变了。

因此,后面讲的字符串拼接、替换、去空格等操作都会得到新的字符串。

public class 字符串变内容不变 {
	// 说明:System.identityHashCode(Onject):不同对象,此结果必不同
	public static void main(String[] args) {
		String s1 = "黄袍怪";
		System.out.println(System.identityHashCode(s1));
		s1 = "奎木狼";// s1保存的是地址值,之前的字符串对象没变,只是s1的地址变了
		System.out.println(System.identityHashCode(s1));
		String s2 = "黄袍怪";
		System.out.println(System.identityHashCode(s2));
	}
}

字符串提取

char charAt(int index) 从指定位置提取单个字符,该位置由index(索引值)指定
String substring(int index) 提取从index指定的位置开始的部分字符串
String substring(int begin, int end) 提取 begin 和 end 位置之间的部分字符串
String concat(String str) 连接两个字符串,并新建一个包含调用字符串的字符串对象
String replace (char oldChar, char newChar) 将字符串中出现oldChar指定的字符全部替换为newChar指定的字符
replaceAll(String regex, String replacement) 将字符串中匹配regex的字符串全部都替换为replacement指定的字符
String trim() 去除字符串前后的空格,得到的是一个新的字符串。
public class Test提取字符串 {
	public static void main(String[] args) {
		String str1 = "《西游记》中,如来为什么要传经?";
		String str2 = new String("普度众生,一个都不能放过!");
		System.out.println("------------查找--------------");
		int _index = 2;
		char _c = str1.charAt(_index);
		System.out.println("charAt " + _index + ":" + _c);
		System.out.println("------------substring--------------");
		String _subStr = str1.substring(_index);
		System.out.println("substring(" + _index + "):" + _subStr);
		int _begin = 2, _end = 9;
		_subStr = str1.substring(_begin, _end);
		System.out.println("substring(" + _begin + "," + _end + "):" + _subStr);
		System.out.println("------------拼接--------------");
		System.out.println("concat:" + str1.concat(str2));
		System.out.println("String + String:" + str1 + str2);
		System.out.println("------------替换--------------");
		char oldC = '西', newC = '东';
		String _newString = str1.replace(oldC, newC);
		System.out.println("replace(" + oldC + "," + newC + ") :" + _newString);
		String _tar = "记", _repl = "释厄转";
		_newString = str1.replace(_tar, _repl);
		System.out.println("replace(" + _tar + "," + _repl + "):" + _newString);
		_newString = "tel:10086".replaceAll("\\d", "*");
		System.out.println("replaceAll(数字→*):" + _newString);
		System.out.println("------------去空格--------------");
		String s = " A B C  ".trim();
		System.out.println("trim:【" + s + "】");
	}
}

字符串转为其它格式

byte[] getBytes() 将字符串转为byte数组(即字符串在内存中保存的最原始的二进制形态)
char[] toCharArray() 将字符串转为字符数组,类似于C语言中字符串的保存形式
public class Test字符串转化 {
	public static void main(String[] args) {
		byte[] b;
		b = "Hello Java".getBytes();
		for (int i = 0; i < b.length; i++) {
			System.out.print("【" + b[i] + " " + (char) b[i] + "】");
		}
		System.out.println("-------------------------");
		char[] c;
		c = "Hello Java".toCharArray();
		for (int i = 0; i < c.length; i++) {
			System.out.print(" " + c[i]);
		}
	}
}

字符串转为字符串数组形式(.split(...)):

public class StringSplit {
	public static void main(String[] args) {
		String[] sArr = "01,天魁星,呼保义,宋江".split(",");
		for (String s : sArr) {
			System.out.println(s);
		}
	}
}

StringBuiler和StringBuffer

StringBuiler/StringBuffer表示可变的字符序列(即可变的字符串)

使用“+运算符”连接字符串将创建新的字符串,频繁修改字符串时,一般使用StringBuilerh或StringBuffer类以提高效率。

public class StringB {
	public static void main(String[] args) {
		StringBuilder sb = new StringBuilder();
		sb.append("诸");
		sb.append("葛");
		sb.append("孔");
		sb.append("明");
		System.out.println(sb.toString());
	}
}

解析:String的值是final的,因此字符串底层的存储数组是不变的,每次append都创建对象。

private final char value[];

StringBuilder的值不是final的

char[] value;// 在抽象类AbstractStringBuilder中定义

StringBuilder类
构造方法() 构造方法:创建一个空的StringBuffer对象
构造方法(String str) 构造方法:根据字符串str的内容创建StringBuffer对象
append(x) 添加内容:类型不限
insert(int index, x) 插入:将x插入到索引为index的位置,x可以为任何类型的数据
delete(int start, int end) 删除:从start位置开始,直到end指定的索引位置
deleteCharAt(int index) 删除:index指定的索引处的字符
setCharAt(int index, char ch) 替换:使用ch指定的新值替换 index指定的位置上的字符
replace(int start, int end, String str) 替换:从start指定的位置开始替换,直到 end 指定的位置结束
reverse() 字符序列倒置
length() 长度
String toString() 转换为字符串类型
public class TestStringBuXXX {
	static void m最常用方法() {
		StringBuilder sb = new StringBuilder();
		// 【.length()】
		System.out.println("length():" + sb.length());
		// 【.append()】
		sb.append("ABC,");
		sb.append(true);
		// 【.toString()】
		System.out.println("append():" + sb.toString());
		// 链式编程:append方法返回的就是自身,可以继续append
		sb.append("青狮").append("白象").append("大鹏鸟");
		System.out.println(sb.toString());
		// 【转置:.reverse()】
		sb.reverse();
		System.out.println("转置:" + sb.toString());
		// 【清空:.setLength(0)】
		sb.setLength(0);
		System.out.println("清空:" + sb.toString());
	}
	static void m其它常用方法() {
		StringBuffer sb = new StringBuffer("ABCDEFGHI");
		System.out.println("---插入---");
		sb.insert(2, "123");
		System.out.println("|--insert():" + sb);
		System.out.println("---删除---");
		sb.delete(2, 2 + 3);
		System.out.println("|--delete():" + sb);
		sb.deleteCharAt(2);
		System.out.println("|--deleteCharAt():" + sb);
		System.out.println("---替换---");
		sb.setCharAt(2, '*');
		System.out.println("|--setCharAt():" + sb);
		sb.replace(1, 1 + 4, "#@");
		System.out.println("|--replace():" + sb);
	}
	public static void main(String[] args) {
		m最常用方法();
		m其它常用方法();
	}
}

*StringBuffer和StringBuilder用法上一样,StringBuilder适用于单线程环境,效率较高。

*StringBuffer是JDK 1.5新增的。

例:回文

写一个方法,判断某句话是不是回文(英文不区分大小写)。
如:Able was I ere I saw Elba
如:假似真时真似假

public class 回文 {
	public static void main(String[] args) {
		String s = "Able was I ere I saw Elba";
		check(s);
		s = "假似真是真似假";
		check(s);
	}
	static void check(String s) {
		String sLower = s.toLowerCase();
		System.out.println(sLower);
		StringBuffer s2 = new StringBuffer(sLower);
		s2.reverse();
		System.out.println(s2);
		if (sLower.equals(s2.toString())) {
			System.out.println("是回文");
		} else {
			System.out.println("不是回文");
		}
	}
}

*扩展·MessageFormat

java.text.MessageFormat

import java.text.MessageFormat;
import java.util.Date;

public class TestMessageFormat {
	public static void main(String[] args) {
		String template = "姓名:{0},年龄:{1},时间:{2}";
		String s = MessageFormat.format(template, "陈玄奘", 28, new Date());
		System.out.println(s);
		System.out.printf("姓名:%s,年龄:%d,时间:%s", "陈玄奘", 28, new Date());
	}
}
posted @ 2019-07-13 01:31  虎老狮  阅读(378)  评论(0编辑  收藏  举报