先找了一个最简单的java.lang.Boolean开始解剖。
首先我们剔除所有的方法和静态变量,Boolean的核心代码如下:
public final class Boolean implements java.io.Serializable,Comparable
{
private final boolean value;
}
很明显,凡是成员变量都是final类型的,一定是immutable class,这个Boolean和String一样,一旦构造函数执行完毕,实例的状态就不能再改变了。
Boolean的构造函数有两个:
public Boolean(boolean value) {
this.value = value;
}
public Boolean(String s) {
this(toBoolean(s));
}
都很简单就不多说了。
另外注意到Boolean类实际上只有两种不同状态的实例:一个包装true,一个包装false,Boolean又是immutable class,所以在内存中相同状态的Boolean实例完全可以共享,不必用new创建很多实例。因此Boolean class还提供两个静态变量:
public static final Boolean TRUE = new Boolean(true);
public static final Boolean FALSE = new Boolean(false);
这两个变量在Class Loader装载时就被实例化,并且申明为final,不能再指向其他实例。
提供这两个静态变量是为了让开发者直接使用这两个变量而不是每次都new一个Boolean,这样既节省内存又避免了创建一个新实例的时间开销。
因此,用
Boolean b = Boolean.TRUE;
比
Boolean b = new Boolean(true);
要好得多。
如果遇到下面的情况:
Boolean b = new Boolean(var);
一定要根据一个boolean变量来创建Boolean实例怎么办?
推荐你使用Boolean提供的静态工厂方法:
Boolean b = Boolean.valueOf(var);
这样就可以避免创建新的实例,不信看看valueOf()静态方法:
public static Boolean valueOf(boolean b) {
return (b ? TRUE : FALSE);
}
这个静态工厂方法返回的仍然是两个静态变量TRUE和FALSE之一,而不是new一个Boolean出来。虽然Boolean非常简单,占用的内存也很少,但是一个复杂的类用new创建实例的开销可能非常大,而且,使用工厂方法可以方便的实现缓存实例,这对客户端是透明的。所以,能用工厂方法就不要用 new。
和Boolean只有两种状态不同,Integer也是immutable class,但是状态上亿种,不可能用静态实例缓存所有状态。不过,SUN的工程师还是作了一点优化,Integer类缓存了-128到127这256个状态的Integer,如果使用Integer.valueOf(int i),传入的int范围正好在此内,就返回静态实例。
hashCode()方法很奇怪,两种Boolean的hash code分别是1231和1237。估计写Boolean.java的人对这两个数字有特别偏好:
public int hashCode() {
return value ? 1231 : 1237;
}
equals()方法也很简单,只有Boolean类型的Object并且value相等才返true:
public boolean equals(Object obj) {
if (obj instanceof Boolean) {
return value == ((Boolean)obj).booleanValue();
}
return false;
}
顺便提一句:很多人写equals()总是在第一行写:
if (obj==null) return false;
其实完全没有必要,因为如果obj==null,下一行的
if (obj instanceof Type)
就肯定返回false,因为(null instanceof AnyType) = false。
详细内容请参考《Effective Java》第7条:Obey the general contract when overriding equals。
其他的方法如toString()就更简单了,只要稍微熟悉java的程序员相信都能写出来,我就不多说了。
★ 总结 ★
1.如果一个类只有有限的几种状态,考虑用几个final的静态变量来表示不同状态的实例。
例如编写一个Weekday类,状态只有7个,就不要让用户写new Weekday(1),直接提供Weekday.MONDAY即可。
2.要防止用户使用new生成实例,就取消public构造函数,用户要获得静态实例的引用有两个方法:如果申明public static var就可以直接访问,比如Boolean.TRUE,
第二个方法是通过静态工厂方法:Boolean.valueOf(?)
3.如果不提供public构造函数,让用户只能通过上面的方法获得静态变量的引用,还可以大大简化equals()方法:
public boolean equals(Object obj) {
return this==obj;
}
可以直接用==比较引用,绝对没有问题,而且效率最高。
4. 为什么JDK的Boolean没有实现上面第3点?因为那两个static变量TRUE和FALSE是在jdk 1.2以后才有的,由于前面的版本已经把构造函数申明为public,所以为了保持客户端代码能够不修改也在后面的版本中运行,只好继续提供public 构造函数。
分享到:
相关推荐
允许JavaScript代码直接使用Java的类库,例如JavaScript中的数字可以直接映射为java.lang.Double,字符串映射为java.lang.String,布尔值映射为java.lang.Boolean等等。这种设计减少了不必要的类型转换开销,也使得...
很多java开发的小伙伴都会阅读jdk源码,然而确不知道应该从哪读起。以下为小编整理的通常所需阅读的源码范围。 标题为包名,后面序号为优先级1-4,优先级递减 1、java.lang 1) Object 1 2) String 1 3) ...
JDK1.8源码分析 引入原始过程中的注意事项 JDK1.8对应JDK版本下载: 码:49wi 原始码在src目录下 以下两个类手动添加的,解决编译过程中该包的丢失 sun.font.FontConfigManager sun.awt.UNIXToolkit 其中:1.请...
jdk1.8-source-analysis JDK1.8源码分析引入原始过程中的注意事项JDK1.8对应JDK版本下载: 码:49wi原始码在src目录下以下两个类手动添加的,解决编译过程中该包的丢失sun.font.FontConfigManager sun.awt....
jdk1.8-source-analysis JDK1.8源码分析引入原始过程中的注意事项JDK1.8对应JDK版本下载: 码:49wi原始码在src目录下以下两个类手动添加的,解决编译过程中该包的丢失sun.font.FontConfigManager sun.awt....
很多java开发的小伙伴都会阅读jdk源码,然而确不知道应该从哪读起。以下为小编整理的通常所需阅读的源码范围。 标题为包名,后面序号为优先级1-4,优先级递减 1、java.lang 1) Object 1 2) String 1 3) ...
源码分析方面,我们可以关注`java.lang.ClassLoader`类。`loadClass(String name, boolean resolve)`方法是核心,其中`name`参数为全限定类名,`resolve`表示是否立即解析。在`loadClass()`内部,它会先尝试调用`...
- 常用包介绍:`java.util`(集合框架)、`java.io`(输入输出流)、`java.lang`(基础类)等。 - 常用类详解:`String`类的不可变性、`ArrayList`的扩容机制等。 - **第三方库集成**: - 常见库推荐:Apache ...
java.lang.String类是final类型的,因此不可以继承这个类、不能修改这个类。为了提高效率节省空间,我们应该用StringBuffer类 6、int 和 Integer 有什么区别 Java 提供两种不同的类型:引用类型和原始类型(或...
在进行安卓逆向工程之前,首先需要确保计算机上已经正确安装了Java JDK,并且设置了相应的环境变量。可以通过命令`java -version`来测试是否安装成功。 #### 二、初识APK、Dalvik字节码以及Smali ##### 1. APK是...