【Rookie初学Java】
String
终于来到String这一关,想逃是逃不掉的,这一节的博客我将会持续更新,将所学不断放进博客里面
1 关于final的一些认识
final:最终的
String是被final修饰的类,不能被继承;
final修饰方法,则该方法不能被子类继承;
final修饰成员变量,则该变量不能二次附值,所谓二次附值是指除了在对象创建的时候进行一次附值,之后不能再次被更改
当看到String的源码时,我才明白,被final修饰的成员属性的不可更改并不如之前所想,例如这一段源码
构造方法是可以给附初值的,后来仔细一想,确实如此,例如一个人的血型,他在诞生前是不确定的(如果你要说基因决定那当我没说…)排除掉遗传因素:在很久以前的古代,一个人在出生前血型不确定,当他诞生时,他的这一项属性也将随之确定,并且不能修改。
其实final修饰的成员变量是在哪里附值呢?
final int id ;
public Dog(){
this.id = 10;
}
其实他的效果等价于
final int id = 10;
2 String与char数组
-
从java的特性来看,String与数组有本质的区别,String是一个类,其中除了字符数组还有其他的属性,作为一个对象,char[] 中是字符型常量,而String的对象除了字符,还有其他的属性。
-
String对象的之间可以用运算符“+”而char不行(合并操作)
3 String中的一些方法
1 toCharArry()
String 与char数组之间的转换
将String对象的方法toString每一个字符都放到一个数组中。
String ——> char[]
public class Main
{
public static void main(String[] args)
{
String s="Java";
char[] array=s.toCharArray();
System.out.println(array);
}
}
/*
结果:Java
*/
char[] ——> String
char[] chars=new char[]{'J','a','v','a'};
String s = new String(chars);
System.out.println(s);
/*
结果:Java
*/
2 indexOf(String str):
获取字符串str第一次在String中出现的索引类似于c中的strstr()
public class Test05 {
public static void main(String[] args) {
String s="dasjdkkadoowkfskadf";
System.out.println(s.indexOf('a'));
}
}
/*
运行结果:1
*/
plus版
public class Test05 {
public static void main(String[] args) {
String s="dasjdkkadoowkfskadf";
System.out.println(s.indexOf('a',2));//从下标2开始第一次出现'a'的索引
}
}
/*
运行结果:7
*/
3 equals()
String与其他类型不同的地方,equals()用来判断两个String对象是否相等,不是判断地址,判断的是每一个字符。原因是啥?因为相同的String内容可能会放在常量池或者堆中的不同位置。
String s="h";
String s1=new String("h");
System.out.println(s==s1);
/*
false
*/
4 toLowerCase()&&toUpperCase()
功能不做演示了,主要的作用就是将String转为大小写String
5 split()
这个方法挺好用的,可以将字符串根据不同的符号进行分割,返回一个个String对象
我们可以用String数组来接
public static String[] getname() {
System.out.println("请输入所有学生名字:");
String string = scanner.nextLine();
String[] splitname = string.split(" ");
namelength=splitname.length;
return splitname;
}
例如这个方法,将输入的字符串分割开来存进splitname
4 StringBuffer与StringBuild
StringBuffer与StringBuild底层是用一个数组来存储字符串的值,并且数组的默认长度为16,并且可以随着append不断扩容。
- 在使用StringBuffer()时,
new
关键字会在堆内存空间中开辟一块内存地址为连续的且空间大小为16个char大小的数组,而当使用有参构造时晖根据所给字符串的大小加上16作为临时存放字符串的空间 - StringBuild与StringBuffer底层都是靠数组实现的,他们不同的地方就在涉及到多线程的时候有所不同,StringBuffer线程更加安全,而StringBuild所有线程都可以使用,因此StringBuffer的效率也就没有StringBuild高了
关于String与String Buffer与StringBuild
- String对象被创建之后不能被修改,大家可能会问
public class Main {
public static void main(String[] args) {
String s="h";
System.out.println(s);
s="n";
System.out.println(s);
}
}
这种情况难道不是能改了对象吗?
他真的是改变了对象的值吗?No,其实是让s这个引用指向了常量池“n”所在的位置,这还是原来那个对象吗?其实它本质上来看和这段代码等价。
public class Main {
public static void main(String[] args) {
String s="h";
System.out.println(s);
s=new String("n");
System.out.println(s);
}
}
我们在实际开发中最好不要使用构造方法来创建对象,为什么呢?原因就是在类加载的时候JVM已经将你的字面量放在常量池中了,这时候我们可以直接将引用指向常量池中,如果再用构造方法来在堆中创建对象不久浪费了内存吗?
-
我们在实际开发中如果一开始不确定String的内容,我们不要随便用String的对象,我们最好使用StringBuffer与StringBuild来创建对象,避免浪费内存。
造方法来创建对象,为什么呢?原因就是在类加载的时候JVM已经将你的字面量放在常量池中了,这时候我们可以直接将引用指向常量池中,如果再用构造方法来在堆中创建对象不久浪费了内存吗? -
我们在实际开发中如果一开始不确定String的内容,我们不要随便用String的对象,我们最好使用StringBuffer与StringBuild来创建对象,避免浪费内存。
有一个结论:
Java数组末尾没有‘\0’