一.String
1.面试题
/*
String s = new String("hello");
问:如上语句创建了几个对象?
答:2个
第一个:字面量"hello"
---java会创建一个String对象表示字面量"hello",并将其存入常量池
第二个:new String()
---new String()时会再创建一个字符串对象,并引用hello字符串的内容
*/
String s = new String("hello");
String s1 = "hello";
System.out.println("s:"+s); //hello
System.out.println("s1:"+s1); //hello
System.out.println(s==s1); //false,==比较的是地址是否相同
//字符串实际开发中比较相等一般都是比较字符串的内容
//因此我们需要使用equals()方法来比较两个字符串的内容
System.out.println(s.equals(s1)); //equals()比较的是内容是否相同
2.String常用方法:
2.1 length():获取字符串的长度(字符个数)
String str = "我爱Java!";
int len = str.length(); //获取str的长度
System.out.println(len); //7
2.2 trim():去除当前字符串两边的空白字符
String str = " hello world ";
System.out.println(str); // hello world
str = str.trim(); //去除当前字符串两边的空白字符
System.out.println(str); //hello world
2.3 toUpperCase()和toLowerCase():将当前字符串中的英文部分转为全大写/全小写
String str = "我爱Java!";
String upper = str.toUpperCase(); //将str中英文部分转为全大写
System.out.println(upper); //我爱JAVA!
String lower = str.toLowerCase(); //将str中英文部分转为全小写
System.out.println(lower); //我爱java!
2.4 startsWith(String str)和endsWith(String str):判断当前字符串是否是以给定的字符串开始/结尾的
String str = "thinking in java";
boolean starts = str.startsWith("think"); //判断str是否是以think开头的
System.out.println("starts:"+starts); //true
boolean ends = str.endsWith(".png"); //判断str是否是以.png结尾的
System.out.println("ends:"+ends); //false
2.5 charAt():返回当前字符串指定位置上的字符
// 0123456789012345
String str = "thinking in java";
char c = str.charAt(9); //获取位置9所对应的字符
System.out.println(c); //i
2.6 indexOf()和lastIndexOf():检索给定字符串在当前字符串中的开始位置
// 111111
// 0123456789012345
String str = "thinking in java";
int index = str.indexOf("in"); //检索in在字符串str中的开始位置
System.out.println(index); //2
index = str.indexOf("in",3); //从下标为3的位置开始找in第一次出现的位置
System.out.println(index); //5
index = str.indexOf("IN"); //当前字符串不包含IN,所以返回-1
System.out.println(index); //-1
index = str.lastIndexOf("in"); //找in最后一次出现的位置
System.out.println(index); //9
2.7 substring():截取当前字符串中指定范围内的字符串
// 1
// 01234567890
String str = "www.tedu.cn";
String name = str.substring(4,8); //截到下标4到7范围的字符串
System.out.println(name); //tedu
name = str.substring(4); //从下标4开始一直截到末尾
System.out.println(name); //tedu.cn
2.8 String的静态方法valueOf():将其它数据类型转换为String
int a = 123;
String s1 = String.valueOf(a); //将int型变量a转换为String类型并赋值给s1
System.out.println(s1); //123---字符串类型
double d = 123.456;
String s2 = String.valueOf(d); //将double型变量d转换为String类型并赋值给s2
System.out.println(s2); //123.456----字符串类型
String s3 = a+""; //任何内容和字符串连接的结果都是字符串,效率低(下节课讲)
System.out.println(s3); //123---字符串类型
3. StringBuilder类:
3.1 由于String是不变对象,每次修改内容要创建新对象,因此String不适合做频繁修改操作,为了解决这个问题,java提供了StringBuilder类。
3.2 StringBuilder是专门用于修改字符串的一个API,内部维护一个可变的char数组,修改都是在这个数组上进行的,修改速度、性能优秀,并且提供了修改字符串的常见的方法:增、删、改、插
//String不适合频繁修改内容
String s = "a";
for(int i=0;i<10000000;i++){
s = s+i;
}
System.out.println("执行完毕");
//用StringBuilder可以提高修改字符串的性能
StringBuilder builder = new StringBuilder("a");
for(int i=0;i<10000000;i++){
builder.append(i);
}
System.out.println("执行完毕");
4. StringBuilder常用方法
- append():追加内容
- replace():替换部分内容
- delete():删除部分内容
- insert():插入内容
- reverse():翻转内容
String str = "好好学习java";
//复制str中的内容到builder中-----好好学习java
StringBuilder builder = new StringBuilder(str);
//append():追加内容
builder.append(",为了找个好工作!");
System.out.println(builder); //好好学习java,为了找个好工作!
//replace():替换部分内容
builder.replace(9,16,"就是为了改变世界"); //替换下标9到15的
System.out.println(builder); //好好学习java,就是为了改变世界!
//delete():删除部分内容
builder.delete(0,8); //删除下标0到7的
System.out.println(builder); //,就是为了改变世界!
//insert():插入操作
builder.insert(0,"活着"); //从下标0的位置插入
System.out.println(builder); //活着,就是为了改变世界!
//reverse():翻转
builder.reverse(); //翻转内容
System.out.println(builder); //!界世变改了为是就,着活
补充:
1.StringBuilder和StringBuffer:
- StringBuffer:是线程安全的,同步处理的,性能稍慢
- StringBuilder:非线程安全的,并发处理的,性能稍快
2.getter/setter:
class Point{ //点
private int x;
private int y;
public int getX(){ //getter获取值
return x;
}
public void setX(int x){ //setter设置值
this.x = x;
}
public int getY(){
return y;
}
public void setY(int y){
this.y = y;
}
}
//getter和setter的演示
public class GetterSetterDemo {
public static void main(String[] args) {
Point p = new Point();
p.setX(100); //赋值
p.setY(200); //赋值
System.out.println(p.getX()+","+p.getY()); //取值
}
}
3.对字符串进行访问就用String型,如果对字符串进行增删改插则转换为StringBuilder类型
4.简单运用
public class LocalTest {
public static void main(String[] args) {
String name1 = "www.tedu.cn";
String name2 = "www.tedu.com.cn";
String name3 = "http://www.google.com";
String str1 = getName(name1);
System.out.println(str1);
String str2 = getName(name2);
System.out.println(str2);
String str3 = getName(name3);
System.out.println(str3);
}
//获取给定网址的域名 line:网址 返回域名
public static String getName(String line){
int start = line.indexOf(".")+1;
int end = line.indexOf(".",start);
return line.substring(start,end); //含start,不含end。
}
}
1. 正则表达式:
- 正则表达式是用来描述字符串内容格式,使用它通常用来匹配一个字符串的内容是否符合要求
- 正则表达式的语法:-----------了解、不用纠结、不用深入研究
[]:表示一个字符,该字符可以是[]中指定的内容
例如:
[abc]:这个字符可以是a或b或c
[a-z]:表示任意一个小写字母
[a-zA-Z]:表示任意一个字母
[a-zA-Z0-9_]:表示任意一个数字字母下划线
[^abc]:该字符只要不是a或b或c
预定义字符:
.:表示任意一个字符,没有范围限制
\d:表示任意一个数字,等同于[0-9]
\w:表示任意一个单词字符,等同于[a-zA-Z0-9_]
\s:表示任意一个空白字符
\D:表示不是数字
\W:不是单词字符
\S:不是空白字符
量词:
?:表示前面的内容出现0-1次
例如: [abc]? 可以匹配:a 或 b 或 c 或什么也不写
+:表示前面的内容最少出现1次
例如: [abc]+ 可以匹配:b或aaaaaaaaaa...或abcabcbabcbabcbabcbabbabab....
但是不能匹配:什么都不写 或 abcfdfsbbaqbb34bbwer...
*:表示前面的内容出现任意次(0-多次)---匹配内容与+一致,只是可以一次都不写
例如: [abc]* 可以匹配:b或aaaaaaaaaa...或abcabcbabcbabcbabcbabbabab....或什么也不写
但是不能匹配:abcfdfsbbaqbb34bbwer...
{n}:表示前面的内容出现n次
例如: [abc]{3} 可以匹配:aaa 或 bbb 或 aab 或abc 或bbc
但是不能匹配: aaaa 或 aad
{n,m}:表示前面的内容出现最少n次最多m次
例如: [abc]{3,5} 可以匹配:aaa 或 abcab 或者 abcc
但是不能匹配:aaaaaa 或 aabbd
{n,}:表示前面的内容出现n次以上(含n次)
例如: [abc]{3,} 可以匹配:aaa 或 aaaaa.... 或 abcbabbcbabcbabcba....
但是不能匹配:aa 或 abbdaw...
()用于分组,是将括号内的内容看做是一个整体
例如: (abc){3} 表示abc整体出现3次. 可以匹配abcabcabc
但是不能匹配aaa 或abcabc
(abc|def){3}表示abc或def整体出现3次.
可以匹配: abcabcabc 或 defdefdef 或 abcdefabc
但是不能匹配abcdef 或abcdfbdef
2.String支持与正则表达式相关的方法:
方法1: matches():使用给定的正则表达式验证当前字符串的格式是否符合要求
/*
邮箱的正则表达式:
[a-zA-Z0-9_]+@[a-zA-Z0-9]+(\.[a-zA-Z]+)+
*/
String email = "wangkj@tedu.cn";
String regex = "[a-zA-Z0-9_]+@[a-zA-Z0-9]+(\\.[a-zA-Z]+)+";
boolean match = email.matches(regex);
if(match){
System.out.println("是正确的邮箱");
}else{
System.out.println("不是正确的邮箱");
}
方法2: split():将当前字符串按照满足正则表达式的部分进行拆分
String line = "abc123def456ghi";
String[] data = line.split("[0-9]+"); //按数字拆分
System.out.println(Arrays.toString(data)); //将data数组按照字符串的格式输出
line = "123,456,789,482";
data = line.split(","); //按逗号拆分
System.out.println(Arrays.toString(data));
line = "123.456.789.482"; //练习+下课--------11:35继续
data = line.split("\\."); //按点拆分
System.out.println(Arrays.toString(data));
//最开始就是可拆分项(.),那么数组中的第1个元素为一个空字符串------""
//如果连续两个(两个以上)可拆分项,它们中间也会拆出一个空字符串-----""
//如果末尾连续多个可拆分项,那么拆出的空字符串被忽略
line = ".123.456..789.482.......";
data = line.split("\\.");
System.out.println(Arrays.toString(data));
方法3: replaceAll():将当前字符串中满足正则表达式的部分替换为给定的字符串
String line = "abc123def456ghi";
line = line.replaceAll("[0-9]+","#NUMBER#"); //将数字部分替换为#NUMBER#
System.out.println(line);
3.Object:对象/东西
1. 是所有类的鼻祖,所有类都直接或间接继承了Object,万物皆对象,为了多态
2. Object中有几个经常被派生类重写的方法: toString()和equals()
2.1 调用toString()时默认返回: 类的全称@地址,没有参考意义,所以常常重写toString()返回具体数据
2.2 调用equals()时默认比较的还是==(即比较地址),没有参考意义,所以常常重写equals()来比较具体的属性值
需要注意:java提供的类都已经重写equals()了(String、StringBuilder),但我们自己定义的类必须自己重写equals()
派生类重写equals()的基本规则:
1)两个对象必须是同一个类型,若类型不同则返回false
public class GetterSetterDemo {
public static void main(String[] args) {
String s = "abc";
StringBuilder ss = new StringBuilder("abc");
boolean a = s.equals(ss);
System.out.println(a);
}
}
2)若参数对象为null,则返回false
3)原则上两个对象要比较对象的属性是否是相同
import java.util.Objects;
/** 测试常常被派生类重写的Object中的相关方法 */
public class Point {
private int x;
private int y;
@Override
public String toString() {
return "Point{" +
"x=" + x +
", y=" + y +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Point point = (Point) o;
return x == point.x && y == point.y;
}
@Override
public int hashCode() {
return Objects.hash(x, y);
}
public Point(int x, int y) {
this.x = x;
this.y = y;
}
public int getX() {
return x;
}
public void setX(int x) {
this.x = x;
}
public int getY() {
return y;
}
public void setY(int y) {
this.y = y;
}
}
public class ObjectDemo {
public static void main(String[] args) {
//演示重写toString()
Point p = new Point(1,2);
//输出引用对象时默认调用toString()
//相当于System.out.println(p.toString());
System.out.println(p);
//字符串连接时将默认调用对象的toString()
//相当于String str = "这是个点:"+p.toString();
String str = "这是个点:"+p;
System.out.println(str);
//演示重写equals()
//Object类另一个常常被派生类重写的方法:equals()
Point p1 = new Point(1,2);
Point p2 = new Point(1,2);
System.out.println(p1==p2); //false,==比较的地址
System.out.println(p1.equals(p2)); //true,因为重写equals()中比较的是x和y
}
}
public class ObjectDemo {
public static void main(String[] args) {
Point p = new Point(1,2);
/*
point类若不重写object类的tostring()方法,则使用Object中定义的toString(),方法的返回字符串格式为:类的全称@地址
但通常这个返回结果对我们开发没有帮助,因此需要重写toString()方法。
*/
//输出引用对象时默认调用Object中的toString()方法
//System.out.println(p); //默认p.toString(),此时是父类的toString
System.out.println(p);//默认p.toString(),此时是point中重写的的toString0
//字符串连接时默认调用对象的toString()
//相当于String str = "这是个点:"+p.tostring()
String str = "这是个点:"+p;
System.out.println(str);
Point p1 = new Point(1,2);
Point p2 = new Point(1,2);
//object中的equals()内部还是使用==来比较地址
//因此派生类在使用是要想比较内容,需要重写equals()
//若派生类重写了equals(),则调用重写之后的
System.out.println(p1==p2);
System.out.println(p1.equals(p2));
//java的类都重写了equals()方法,我们自己定义的类必须自己重写equals()
//例string和stringbuilder都已经重写了equals()
}
}
4.包装类:
4.1 java定义了8个包装类,目的是为了解决基本类型不能直接参与面向对象开发的问题,使得基本类型可以通过包装类的实例以对象的方式存在
4.2 包括:Integer、Character、Byte、Short、Long、Float、Double、Boolean。其中Character和Boolean是直接继承自Object的,而其余6个包装类都继承自java.lang.Number
4.3 JDK1.5推出了一个新的特性:自动拆装箱,该特性是编译器认可以,当编译器编译时若发现有基本类型与包装类型相互赋值时,将会自动补充代码来完成他们的转换工作,这个过程称为自动拆装箱
//演示自动拆装箱
//触发了自动装箱特性
Integer i1 = 5; //会被编译为: Integer i1 = Integer.valueOf(5);
//触发了自动拆箱特性
int ii = i1; //会被编译为: int ii = i1.intValue();
//演示包装类的实际操作:
//1)可以通过包装类来得到基本类型的取值范围:
int max = Integer.MAX_VALUE; //获取int的最大值
int min = Integer.MIN_VALUE; //获取int的最小值
System.out.println("int的最大值为:"+max);
System.out.println("int的最小值为:"+min);
long lMax = Long.MAX_VALUE; //获取long的最大值
long lMin = Long.MIN_VALUE; //获取long的最小值
System.out.println("long的最大值为:"+lMax);
System.out.println("long的最小值为:"+lMin);
//2)包装类可以将字符串转换为对应的基本类型
// 前提是该字符串正确表达了基本类型的值
// 若不能正确表达,则发生NumberFormatException数字转换异常
String str = "123";
int num = Integer.parseInt(str); //将字符串str转换为int类型
System.out.println(num); //123
str = "123.456";
double dou = Double.parseDouble(str); //将字符串str转换为double类型
System.out.println(dou); //123.456
//double不能转int,,int可以转double
1.给Integer型变量赋值的三种方式:
Integer i1 = Integer.valueOf(5);
Integer in = new Integer(5);
Integer i1 = 5;