一 关于字符串的陷阱
1 JVM对字符串的处理
在此之前,先对于java中对象的四种创建方式有个了解:
(1)通过new操作符;
(2)通过Class对象的newInstance()方法来调用构造器创建;
(3)通过java的反序列化机制从IO流中恢复java对象;
(4)通过java对象提供的clone()方法复制一个新的java对象。
言归正传:
(1)在java中,对于字符串直接量,JVM会使用一个缓冲池来保存它们,当是第一次来使用它时,JVM会将它放入缓冲池中进行保存;当对它进行再次访问时,无须重新创建一个新的字符串,而是直接让引用变量指向缓存在缓冲池中已存在的字符串。
(2)字符串是在编译时确定其值还是在运行时确定:对于字符串直接量,整数直接量,没有变量参与,则由JVM在编译时就可确定;对于程序若使用变量或者方法的,则由JVM在运行时确定。
(3)对于(2)中的特例:当用final关键字声明变量,进行字符串连接运算执行“宏替换”,那么JVM同样可在编译时就确定字符串连接表达式中的值。
2 不可变的字符串
java中String类是一个典型的不可变类,当一个String对象创建完成后,该String类里包含的字符序列就固定下来,就无法发生改变。
运用情景:
若程序需要一个字符序列会发生改变的字符串,那么该怎么办呢?这时就不由想到StringBuilder和StringBuffer。但是这二者之中该如何选择?实际运用中,通常会优先考虑StringBuilder。原因:StringBuffer是线程安全的,那么其中的方法都包含synchronized关键字,由此必会降低方法的执行效率。 因此,在没有多线程的环境下,应该优先使用StringBuilder类。
3 字符串的比较
(1)
==:用于比较两个字符串是否相同;
(2)equals()方法:判断字符串所包含的字符串序列是否相同;
(3)compareTo()方法:用于比较两个字符串的大小,当两字符串所包含的字符序列相同时,返回值0. 如何判断?
判断方法:先将两个字符串左对齐,然后从左向右依次比较每个字符,包含较大字符的字符串则较大。如:“abcx”和“ax”比较,后者大。
4 表达式类型的陷阱
首先,对于强类型语言的两个基本特征的阐述:
(1)所有变量必须先得声明,而后才能使用;
(2)一旦某个变量的数据类型确定下来,那么此变量将永远只能接受该类型的值。
4.1 表达式类型的自动提升
依据:基本类型的等级图
char
->int-》long->float->double
byte-》short
4.2复合赋值运算符的陷阱
注意点:隐式类型转换
例如: short st = 5;
st +=10;//相当于st = (short)st + 5,不会发生错误;
st +=90000;//发生错误
为了避免这种潜在的错误,应注意以下几点:
1 将其运用于byte,short或者char等类型的变量时;
2运用于int类型的变量,而表达式右侧是long,float或者double类型的值;
3运用于float,而右侧是double类型。
5 ClassCastException异常发生的原因:集合元素中的实际类型与集合所带的泛型信息不匹配。
6 多线程的陷阱
6.1不要调用run()方法
三种方式来创建,启动多线程:
(1)继承Thread类,override run()方法;
(2)实现Runnable接口,override run()方法;
(3)实现Callable接口, overri call()方法。
其中,第一种效果最差,原因如下:
(1)继承了Thread类,无法继承其他父类;
(2)由于每条线程都是Thread子类的实例,因此可以将多条线程的执行流代码与业务数据分离。
6.2静态的同步方法
java语法规定:任何线程进入同步方法,同步代码块之前,必须先获得同步方法,同步代码块对应的同步监视器。
对于同步代码块:须显示指定同步监视器;对于同步非静态方法而言,为this--调用该方法的java对象;对于静态的同步方法而言,为类本身。
6.3线程安全类
(1)该类的对象可以被多个线程安全地访问;
(2)每个线程调用该对象的任意方法之后都将得到正确结果;
(3)每个线程调用该对象的任意方法之后,该对象状态依然保持合理状态。