字符串
字符串的概述
String的路径java.lang不需要导包
API文档当中说:Java 程序中的所有字符串字面值(如 “abc” )都作为此类的实例实现。
其实就是说:程序当中所有的双引号字符串,都是String类的对象。(就算没有new,也照样是。)
字符串的特点
- 字符串的内容永不可变,常量。【重点】
- 正是因为字符串不可改变,所以字符串是可以共享使用的。
- 字符串效果上相当于是char[]字符数组,但是底层原理是byte[]字节数组。
字符串的构造方法和直接创建
创建字符串的常见3+1种方式。
三种构造方法:
public String():创建一个空白字符串,不含有任何内容。""
public String(char[] array):根据字符数组的内容,来创建对应的字符串。
public String(byte[] array):根据字节数组的内容,来创建对应的字符串。
一种直接创建:
String str = “Hello”; // 右边直接用双引号
注意:对于字符串不需要new也能创建对象,直接写上双引号,就是字符串对象,JVM帮我们new了
// 使用空参构造
String str1 = new String(); // 小括号留空,说明字符串什么内容都没有。
System.out.println("第1个字符串:" + str1); //空
// 根据字符数组创建字符串
char[] charArray = { 'A', 'B', 'C' };
String str2 = new String(charArray);
System.out.println("第2个字符串:" + str2); //ABC
// 根据字节数组创建字符串
byte[] byteArray = { 97, 98, 99 };
String str3 = new String(byteArray);
System.out.println("第3个字符串:" + str3); //abc
// 直接创建
String str4 = "Hello";
System.out.println("第4个字符串:" + str4); //Hello
字符串的常量池
程序当中直接写上的双引号字符串,就在字符串常量池中。只有双引号的字符串才在字符串池中。
对于基本类型来说,==是进行数值的比较。
对于引用类型来说,==是进行【地址值】的比较。
String str1 = "abc"; //直接创建
String str2 = "abc"; //直接创建
char[] charArray = {'a', 'b', 'c'};
String str3 = new String(charArray); //使用字符数组创建
System.out.println(str1 == str2); // true 都在字符串常量池地址相同
System.out.println(str1 == str3); // false 字符数组创建不在字符串常量池地址不同
System.out.println(str2 == str3); // false
String str1 变量存储地址
在堆内存中有字符串常量池,有字符串对象,其保存的是字节数组byte[] 的地址。字符串自己的地址0x666被赋值到str1
JVM创建"abc”时,化作一个字节数组byte[],再把字节数组的地址保存到池当中的对象里,最后把池中对象的地址交给str1
String str2 也是在栈中创建变量存放地址
“abc”,双引号直接创建的字符串在常量池中,池中已有指向"abc"字节数组的地址,不会再次创建,直接把池中对象的地址0x666
交给str2
str1,str2重复利用同一个字符串
char[] charArray 在栈中创建变量名称
数组{‘a’,‘b’,‘c’}在堆中内存创建,charArray保存数组在堆内存的地址,即charArry指向数组堆内存的位置
String str3 在栈中创建变量名称,new String(charArray) ,【只有双引号跟变量池有关并且与new方法无关】
charArray作为构造方法的参数变量,先char型数组转换为byte[]数组,然后new一个字符串对象String存放转化的byte[]数组的地址,即字符串对象指向字节数组。
最后将字符串对象的地址0x999赋值给栈中的变量str3,str3指向字符串对象。
强调
1对于引用类型 ==进行的是地址值的比较
2双引号直接写的字符串在常量池当中,new的不在池当中
字符串比较的相关方法
==是进行对象的地址值比较,如果确实需要字符串的内容比较,可以使用两个方法:
(一)public boolean equals(Object obj):参数可以是任何对象,只有参数是一个字符串并且内容相同的才会给true;否则返回false。
注意事项:
1任何对象都能用Object进行接收。
2equals方法具有对称性,也就是a.equals(b)和b.equals(a)效果一样。
3如果比较双方一个常量一个变量,推荐把常量字符串写在前面。 推荐:“abc”.equals(str) 不推荐:str.equals(“abc”) 原因:当变量值为空时,会出现空指针异常报错,所以尽量常量在前。
String str1 = "Hello";
String str2 = "Hello";
System.out.println(str1.equals(str2)); // true
String str5 = null;
System.out.println("abc".equals(str5)); // 推荐:false
System.out.println(str5.equals("abc")); // 不推荐:报错,空指针异常NullPointerException
(二)public boolean equalsIgnoreCase(String str):忽略大小写,进行内容比较。
String strA = "Java";
String strB = "java";
System.out.println(strA.equalsIgnoreCase(strB)); // true,忽略大小写
字符串获取相关方法
public int length():获取字符串当中含有的字符个数,拿到字符串长度。
public String concat(String str):将当前字符串和参数字符串拼接成为返回值新的字符串。
public char charAt(int index):获取指定索引位置的单个字符。(索引从0开始。)
public int indexOf(String str):查找参数字符串在本字符串当中首次出现的索引位置,如果没有返回-1值。
// 获取字符串的长度
int length = "asdasfeutrvauevbueyvb".length();
System.out.println("字符串的长度是:" + length);
// 拼接字符串
String str1 = "Hello";
String str2 = "World";
String str3 = str1.concat(str2);
System.out.println(str1); // Hello,原封不动
System.out.println(str2); // World,原封不动
System.out.println(str3); // HelloWorld,新的字符串
// 获取指定索引位置的单个字符
char ch = "Hello".charAt(1);
System.out.println("在1号索引位置的字符是:" + ch);
// 查找参数字符串在本来字符串当中出现的第一次索引位置
// 如果根本没有,返回-1值
String original = "HelloWorldHelloWorld";
int index = original.indexOf("llo");
System.out.println("第一次索引值是:" + index); // 2
System.out.println("HelloWorld".indexOf("abc")); // -1
字符串的截取方法
public String substring(int index):截取从参数位置一直到字符串末尾,返回新字符串。
public String substring(int begin, int end):截取从begin开始,一直到end结束,中间的字符串。
备注:[begin,end),包含左边,不包含右边。
String str1 = "HelloWorld";
String str2 = str1.substring(5);
System.out.println(str1); // HelloWorld,原封不动
System.out.println(str2); // World,新字符串
String str3 = str1.substring(4, 7);
System.out.println(str3); // oWo
字符串常量不改变,变得是变量指向的地址
/ 下面这种写法,字符串的内容仍然是没有改变的
// 下面有两个字符串:"Hello","Java"
// strA当中保存的是地址值。
// 本来地址值是Hello的0x666,
// 后来地址值变成了Java的0x999
String strA = "Hello";
System.out.println(strA); // Hello
strA = "Java";
System.out.println(strA); // Java
选中一个单词技巧:
shift+左或右方向键能一直选择
字符串的转换相关方法
String当中与转换相关的常用方法有:
public char[] toCharArray():将当前字符串拆分成为字符数组作为返回值。
public byte[] getBytes():获得当前字符串底层的字节数组。
public String replace(CharSequence oldString, CharSequence newString):
将所有出现的老字符串替换成为新的字符串,返回替换之后的结果新字符串。
备注:CharSequence意思就是说可以接受字符串类型。
// 转换成为字符数组
char[] chars = "Hello".toCharArray();
System.out.println(chars[0]); // H
System.out.println(chars.length); // 5
// 转换成为字节数组
byte[] bytes = "abc".getBytes();
for (int i = 0; i < bytes.length; i++) {
System.out.println(bytes[i]); //97 98 99
}
// 字符串的内容替换
String str1 = "How do you do?";
String str2 = str1.replace("o", "*");
System.out.println(str1); // How do you do?
System.out.println(str2); // H*w d* y*u d*?
进入函数技巧:
按住Ctrl 鼠标点 能进入函数 或者Ctrl+N 输入函数名字也能进入函数
字符串的分割方法
分割字符串的方法:
**public String[] split(String regex):**按照参数的规则,将字符串切分成为若干部分,方法返回值String类型的数组
注意事项:
split方法的参数其实是一个“正则表达式”。
注意:如果按照英文句点“.”进行切分,必须写"\."(两个反斜杠)
String str1 = "aaa,bbb,ccc";
String[] array1 = str1.split(",");
for (int i = 0; i < array1.length; i++) {
System.out.println(array1[i]);
} //aaa bbb ccc
题目:
键盘输入一个字符串,并且统计其中各种字符出现的次数。
种类有:大写字母、小写字母、数字、其他
思路:
- 既然用到键盘输入,肯定是Scanner
- 键盘输入的是字符串,那么:String str = sc.next();
- 定义四个变量,分别代表四种字符各自的出现次数。
- 需要对字符串一个字、一个字检查,String–>char[],方法就是toCharArray()
- 遍历char[]字符数组,对当前字符的种类进行判断,并且用四个变量进行++动作。
- 打印输出四个变量,分别代表四种字符出现次数。
public class Demo07StringCount {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
System.out.println("请输入一个字符串:");
String input = sc.next(); // 获取键盘输入的一个字符串
int countUpper = 0; // 大写字母
int countLower = 0; // 小写字母
int countNumber = 0; // 数字
int countOther = 0; // 其他字符
char[] charArray = input.toCharArray();
for (int i = 0; i < charArray.length; i++) {
char ch = charArray[i]; // 当前单个字符
if ('A' <= ch && ch <= 'Z') {
countUpper++;
} else if ('a' <= ch && ch <= 'z') {
countLower++;
} else if ('0' <= ch && ch <= '9') {
countNumber++;
} else {
countOther++;
}
}
System.out.println("大写字母有:" + countUpper);
System.out.println("小写字母有:" + countLower);
System.out.println("数字有:" + countNumber);
System.out.println("其他字符有:" + countOther);
}
}
静态
static关键字概述
有一个学生类 成员变量 有姓名,年龄,学号,所在教室(特殊)
学生类创造的学生对象 有姓名,年龄,学号,所在教室(特殊),每个学生对象的姓名,年龄,学号各不相同,有各自独立的信息。
但对象的所在教室是相同的(同班),所在教室为共有信息,多个对象共享同一份数据,不必每一个对象都写一份,直接写到类当中保存。
一旦用了static关键字,那么这样的内容不再属于对象自己,而是属于类的,凡是本类的对象,都共享同一份
静态static关键字修饰成员变量
如果一个成员变量使用了static关键字,那么这个变量不再属于对象自己,而是属于所在的类。多个对象共享同一份数据。
学生类
学生类有两个变量用static修饰了,属于学生类,是学生对象的共有数据,所在教室和学号计数器。
只要创建一个对象,调用构造方法,学号计数器就会++然后自动生成学生对象的学号。
对一个对象的类静态变量赋值,其他对象的静态变量也被赋值了
public class Student {
private int id; // 学号
private String name; // 姓名
private int age; // 年龄
static String room; // 所在教室
private static int idCounter = 0; // 学号计数器,每当new了一个新对象的时候,计数器++
public Student() {
this.id = ++idCounter;
}
public Student(String name, int age) {
this.name = name;
this.age = age;
this.id = ++idCounter;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
public static void main(String[] args) {
Student two = new Student("黄蓉", 16);
two.room = "101教室";
System.out.println("姓名:" + two.getName()
+ ",年龄:" + two.getAge() + ",教室:" + two.room
+ ",学号:" + two.getId());
Student one = new Student("郭靖", 19);
System.out.println("姓名:" + one.getName()
+ ",年龄:" + one.getAge() + ",教室:" + one.room
+ ",学号:" + one.getId());
}
两个对象的room属性都是相同的值
静态static关键字修饰成员方法
一旦使用static修饰成员方法,那么这就成为了静态方法。静态方法不属于对象,而是属于类的。
如果没有static关键字,那么必须首先创建对象,然后通过对象才能使用它。
如果有了static关键字,那么不需要创建对象,直接就能通过类名称来使用它。
无论是成员变量,还是成员方法。如果有了static,都推荐使用类名称进行调用。
静态变量:类名称.静态变量
静态方法:类名称.静态方法()
注意事项:
- 静态不能直接访问非静态。
原因:因为在内存当中是【先】有的静态内容,【后】有的非静态内容。
静态方法只能访问静态变量。
“先人不知道后人,但是后人知道先人。” - 静态方法当中不能用this。
原因:this代表当前对象,通过谁调用的方法,谁就是当前对象。
代码解析
MyClass类中有成员变量,静态变量,成员方法,静态方法
Demo02StaticMethod类中用类MyClass类的成员方法:
创建对象obj,才能用对象名.成员方法的方式调用成员方法method(),
Demo02StaticMethod类中用类MyClass类的静态方法:
两种方式,可以用对象名. 也可以用类名. ,为了区别成员方法的调用,推荐使用类名.
Demo02StaticMethod类调用本类中的静态方法,可以省略类名称. 直接采用方法名调用即可 方法名();
Demo02StaticMethod类调用本类中的成员方法,直接用方法名();调用是报错的,必须D用emo02StaticMethod类创建对象才能调用本类的成员方法 对象名.成员方法();
MyClass中的成员方法既能使用成员变量,又能使用静态变量,MyClass中的静态方法只能使用静态变量,并且不能 使用this关键字
public class MyClass {
int num; // 成员变量
static int numStatic; // 静态变量
// 成员方法
public void method() {
System.out.println("这是一个成员方法。");
// 成员方法可以访问成员变量
System.out.println(num);
// 成员方法可以访问静态变量
System.out.println(numStatic);
}
// 静态方法
public static void methodStatic() {
System.out.println("这是一个静态方法。");
// 静态方法可以访问静态变量
System.out.println(numStatic);
// 静态不能直接访问非静态【重点】
// System.out.println(num); // 错误写法!
// 静态方法中不能使用this关键字。
// System.out.println(this); // 错误写法!
}
}
public class Demo02StaticMethod {
public static void main(String[] args) {
MyClass obj = new MyClass(); // 首先创建对象
// 然后才能使用没有static关键字的内容
obj.method();
// 对于静态方法来说,可以通过对象名进行调用,也可以直接通过类名称来调用。
obj.methodStatic(); // 正确,不推荐,这种写法在编译之后也会被javac翻译成为“类名称.静态方法名”
MyClass.methodStatic(); // 正确,推荐
// 对于本类当中的静态方法,可以省略类名称
myMethod();
Demo02StaticMethod.myMethod(); // 完全等效
//对于本类当中的成员方法,还是得创建对象才能调用
Demo02StaticMethod demo= new Demo02StaticMethod()
demo.method();
}
public static void myMethod() {
System.out.println("自己的静态方法!");
}
public void method() {
System.out.println("自己的成员方法!");
}
}
静态static的内存图
方法区中有Student.class,方法区中开辟一空间来存储静态变量,称位静态区。
当根据类名称使用静态变量时,全程和对象没有关系,直接到方法区中找到所在类,再找到静态区取得变量。
只和类有关系
静态代码块
静态代码块的格式是:
public class 类名称 {
static {
// 静态代码块的内容
}
}
特点:当第一次用到本类时,静态代码块执行唯一的一次。
静态内容总是优先于非静态,所以静态代码块比构造方法先执行。
静态代码块的典型用途:
用来一次性地对静态成员变量进行赋值。
代码
public class Person {
static {
System.out.println("静态代码块执行!");
}
public Person() {
System.out.println("构造方法执行!");
}
}
public class Demo04Static {
public static void main(String[] args) {
Person one = new Person();
Person two = new Person();
}
}
结果是
静态代码块执行!
构造方法执行!
构造方法执行!
常用的工具类
数组工具类
Arrays(注意s)
路径java.util
java.util.Arrays是一个与数组相关的工具类,里面提供了大量静态方法,用来实现数组常见的操作(静态的方法不用new,直接类名称Arrays.)。
public static String toString(数组):将参数数组变成字符串(按照默认格式:[元素1, 元素2, 元素3…])
public static void sort(数组):按照默认升序(从小到大)对数组的元素进行排序。
备注:
- 如果是数值,sort默认按照升序从小到大
- 如果是字符串,sort默认按照字母升序
- 如果是自定义的类型,那么这个自定义的类需要有Comparable或者Comparator接口的支持。
代码
int[] intArray = {10, 20, 30};
// 将int[]数组按照默认格式变成字符串
String intStr = Arrays.toString(intArray);
System.out.println(intStr); // [10, 20, 30]
int[] array1 = {2, 1, 3, 10, 6};
Arrays.sort(array1);
System.out.println(Arrays.toString(array1)); // [1, 2, 3, 6, 10]
练习
题目:
请使用Arrays相关的API,将一个随机字符串中的所有字符升序排列,并倒序打印。
public class Demo02ArraysPractise {
public static void main(String[] args) {
String str = "asv76agfqwdfvasdfvjh";
// 如何进行升序排列:sort
// 必须是一个数组,才能用Arrays.sort方法
// String --> 数组,用toCharArray
char[] chars = str.toCharArray();
Arrays.sort(chars); // 对字符数组进行升序排列
// 需要倒序遍历
for (int i = chars.length - 1; i >= 0; i--) {
System.out.println(chars[i]);
}
}
}
如何快速倒序数组技巧
数组名称.forr回车 chars.forr
数学工具类
路径 java.util.Math
java.util.Math类是数学相关的工具类,里面提供了大量的静态方法,完成与数学运算相关的操作。
public static double abs(double num):获取绝对值。有多种重载。
public static double ceil(double num):向上取整。
public static double floor(double num):向下取整。
public static long round(double num):四舍五入,不带小数点。
Math.PI代表近似的圆周率常量(double)。
代码
// 获取绝对值
System.out.println(Math.abs(3.14)); // 3.14
System.out.println(Math.abs(0)); // 0
System.out.println(Math.abs(-2.5)); // 2.5
// 向上取整
System.out.println(Math.ceil(3.9)); // 4.0
System.out.println(Math.ceil(3.1)); // 4.0
System.out.println(Math.ceil(3.0)); // 3.0
// 向下取整,抹零
System.out.println(Math.floor(30.1)); // 30.0
System.out.println(Math.floor(30.9)); // 30.0
System.out.println(Math.floor(31.0)); // 31.0
System.out.println("================");
System.out.println(Math.round(20.4)); // 20
System.out.println(Math.round(10.5)); // 11