String对象用于保存字符串,也就是一组字符序列
常量对象:字符串常量对象是用双引号括起来的字符序列,例如:"小范",”12.34“,”xiaofan“等
字符串的字符使用Unicode字符编码,一个字符(不区分字母还是汉字)占两个字节。

String对象的两种创建方式
第一种:String s1 = "xiaofan";
第二种:String s2 = new String("xiaofan");
说明:
方法一是直接赋值,而方法二是调用构造器。方法一会直接将引用指向常量池中的”xiaofan“,而方法二会先在堆中创建一个空间,在此空间中放入指向常量池"xiaofan"的地址,再让栈空间的对象名指向堆地址。

由于这两种创建方式的不同指向,会出现很多字符串比较是否相等上的问题。
简单案例
package class_string;
public class ClassTest {
public static void main(String[] args) {
String s = "hello"; //常量池中创建hello,指向它
String ss = new String ("hello"); //堆中创建对象,指向常量池中的hello
System.out.println(s.equals(ss)); //比较内容,T
System.out.println(s == ss); //比较地址,F
String a = "abc";
String b = "abc"; // 因为上面a已经在常量池中创建了abc,所以b直接指向b就可以了
System.out.println(a.equals(b)); //比较内容,T
System.out.println(a == b); //比较地址,T
String x = new String("XYZ");
String y = new String("XYZ"); //x、y都在堆中创建了对象,但它们的对象都指向常量池中的XYZ
System.out.println(x.equals(y)); //比较内容,T
System.out.println(x == y); //比较地址,F
System.out.println("============");
String a1 = "abc"; // a1指向 常量池
String b1 =new String("abc");//b1 指向堆
System.out.println(a1.equals(b1)); //T , 比较内容
System.out.println(a1==b1); // F,比较地址
System.out.println(a1==b1.intern()); //T , b1.intern() 指向常量池的”abc”
System.out.println(b1==b1.intern());//F , 对象b1的堆地址和常量池中"abc"的地址比较
/* b1.intern() 方法最终返回的是常量池的地址(对象)
* 当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串
* (用 equals(Object) 方法确定),则返回池中的字符串。
* 否则,将此 String 对象添加到池中,并返回此 String 对象的引用
*/
System.out.println("============");
class Person{
private String name;
}
Person p1 = new Person();
p1.name = "chuannong";
Person p2 = new Person();
p2.name = new String("chuannong");
System.out.println(p1.name.equals(p2.name)); //T
System.out.println(p1.name == p2.name); //F,p1.name 指向常量池,p2.name 指向堆空间
System.out.println(p1.name == "chuannong"); //T
System.out.println(p1.name.intern()); //返回"chuannong"
}
}
字符串的特性
可以从最开始介绍String类的时候看到,它的对象底层是保存在一个final数组中的。
所以对于String对象保存的字符串来所,一个字符串对象一旦分配,其内容就是不可变的。
package class_string;
/**
* String is final
*/
public class ClassTest02 {
public static void main(String[] args) {
//在常量池中创建了两个对象,"abc"、"hello"
String a = "abc";
a = "hello";
//在常量池中创建了一个对象,"abchello",可以理解为,abc与hello并没有被指向引用,所以不需要创建
String b = "abc" + "hello"; //编译器会自动的处理"+",将abc 和 hello 拼接起来
//在常量池中创建了三个对象,"xiao"、"fan"、"xiaofan"
String c = "xiao"; //有c指向"xiao",所以创建
String d = "fan"; //同理
String e = c + d ;
/*
* 因为String对象 底层是放在 private final char value[]; 这里面的
* 所以String是final的,若想要进行拼接或更改字符串,只能是重新在常量池中创建对象
*/
}
}
String类使用
了解了String对象两种声明方式不同,以及字符串特性之后。就又会出现一些对于字符串声明上的问题。
简单案例
package class_string;
public class ClassTest03 {
String str = new String("good");
final char[] ch = { 't', 'e', 's', 't' };
public static void main(String[] args) {
ClassTest03 ex = new ClassTest03();
ex.change(ex.str, ex.ch);
System.out.print(ex.str + " and ");
System.out.println(ex.ch);
}
public void change(String str, char ch[]) {
str = "test";
ch[0] = 'g';
}
}
程序输出:good and gest
案例理解:
-
程序加载时,首先在栈空间创建main方法空间,创建ex对象
-
在ex对象的堆空间中,创建一个String引用str指向另外一个堆空间,此堆空间中放"good"在常量池中的地址
-
在ex对象的堆空间中,创建一个char[]引用ch 指向另外一个堆空间,此堆空间存放 { 't', 'e', 's', 't' }
-
在栈空间中创建change方法空间,传入对象ex的str与ch引用地址
-
str = "test" ,在常量池中创建"test",返回给str,即将change方法中的str的引用改变,不再指向堆空间的地址,而变为指向常量池中的"test"
-
ch[0] = 'g' ,将change方法中的ch[0]该为g,即直接改变堆空间中的ch数组
-
change方法结束,change方法空间销毁,此时 main方法空间中的str并没有改变,ch数组的第一个改变为g,常量池中多了一个"test"。

String类的常用方法
第一部分:
package class_string;
public class StringMethods {
public static void main(String[] args) {
String username = "JOHN";
// 2.equalsIgnoreCase 忽略大小写的判断内容是否相等
if ("john".equalsIgnoreCase(username)) {
System.out.println("Success!");
} else {
System.out.println("Failure!");
}
// 3.length 获取字符的个数,字符串的长度
System.out.println("小范".length()); //输出:2
// 4.indexOf 获取字符在字符串对象中第一次出现的索引,索引从0开始,如果找不到,返回-1
String s1 = "wer@terwe@g";
int index = s1.indexOf('@');
System.out.println(index);//输出:3
// 5.lastIndexOf 获取字符子啊字符串中最后一次出现的索引,索引从0开始,如果找不到,返回-1
String s2 = "wer@terwe@g@";
int index2 = s1.lastIndexOf('@');
System.out.println(index);//输出:3
// 6.substring 截取指定范围的子串
String name = "hello,小范";
System.out.println(name.substring(6));// 截取后面的字符,输出:小范
System.out.println(name.substring(0, 5));//输出:hello
}
}
第二部分:
package class_string;
public class StringMethods02 {
public static void main(String[] args) {
// 1.toUpperCase转换成大写
String s = "heLLo";
System.out.println(s.toUpperCase());
// 2.toLowerCase
System.out.println(s.toLowerCase());
// 3.concat拼接字符串
String s1 = "小范";
s1 = s1.concat("小黄").concat("小雨").concat("together");
System.out.println(s1);
// 4.replace 替换字符串中的字符
String s2 = "小范 and 小黄 小黄 小黄 小黄";
s2 = s2.replace("小黄", "小雨");
System.out.println(s2);
// 5.split 分割字符串, 对于某些分割字符,我们需要 转义比如 | \等
System.out.println("======================");
String poem = "鹅鹅鹅,曲项向天歌,白毛浮绿水,红掌拨清波";
String[] split = poem.split(",");
for (int i = 0; i < split.length; i++) {
System.out.println(split[i]);
}
System.out.println("======================");
String poem2 = "E:\附加项目\附加-project";
String[] split3 = poem2.split("\\");
for (int i = 0; i < split3.length; i++) {
System.out.println(split3[i]);
}
// 6.toCharArray 转换成字符数组
System.out.println("======================");
String s4 = "happy";
char[] chs = s4.toCharArray();
for (int i = 0; i < chs.length; i++) {
System.out.println(chs[i]);
}
// 7.compareTo 比较两个字符串的大小,如果前者大,则返回正数,后者大,则返回负数,如果相等,返回0
System.out.println("======================");
String a = "jchn";
String b = "jack";
System.out.println(a.compareTo(b)); // 返回值是 'c' - 'a' = 2的值
System.out.println("a".compareTo("d")); // 返回值是 'a' - 'd' = -3的值
System.out.println("a".compareTo("a")); // 返回值是 'a' - 'a' = 0的值
// 8.format 格式字符串
/*
* 占位符有: %s 字符串 %c 字符 %d 整型 %.2f 浮点型
*
*/
System.out.println("======================");
String name = "john";
int age = 10;
double score = 98.3 / 3;
char gender = '男';
// String info =
// "我的姓名是"+name+"年龄是"+age+",成绩是"+score+"性别是"+gender+"。希望大家喜欢我!";
String info = String.format("我的姓名是%s年龄是%d,成绩是%.2f性别是%c.希望大家喜欢我!", name, age, score, gender);
System.out.println(info);
}
}
程序输出:
HELLO
hello
小范小黄小雨together
小范 and 小雨 小雨 小雨 小雨
========================
鹅鹅鹅
曲项向天歌
白毛浮绿水
红掌拨清波
========================
E:
附加项目
附加-project
========================
h a p p y
========================
========================
我的姓名是john年龄是10,成绩是32.77性别是男.希望大家喜欢我!
简单应用案例
package class_string;
import java.util.Scanner;
/**
* 判断邮箱是否合法,要求里面必须包含@和. 而且 @ 必须在. 的前面
案例:不使用系统提供的trim 方法,自己写一个myTrim方法,去除字符串两端的空格,
比如 " hello world " => 返回 “hello world”
*/
public class ClassWork01 {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
System.out.println("请输入邮箱:");
String email = scanner.next();
if(legalEmail(myTrim(email))) {
System.out.println("输入正确,你的Email地址为:" + email);
}else {
System.out.println("输入错误!");
}
}
public static String myTrim(String email) {
String temp = "";
for(int i = 0;i<email.length();i++) { //去除前面的空格
if(email.charAt(i) != ' ') {
temp = email.substring(i);
break;
}
}
for(int i = temp.length() - 1;i >= 0;i--) { //去除后面的空格
if(temp.charAt(i) != ' ') {
email = temp.substring(0,i);
break;
}
}
return email;
}
public static boolean legalEmail(String email) {
if(email.indexOf("@") < 0 && email.indexOf(".") < 0) { //没有@和.
return false;
}else {
if(email.indexOf("@") != email.lastIndexOf("@")) { //不止一个@
return false;
}else {
if(email.indexOf("@") > email.indexOf(".")) { //"."不在@之后
return false;
}
}
}
return true;
}
}