书读得多而不思考,你会觉得自己知道的很多。
书读得多而思考,你会觉得自己不懂的越来越多。 ——伏尔泰
三十一、 在接口中不要存在实现代码
接口是一个契约,不仅仅约束着实现者,同时也是一个保证,保证提供的服务(常量、方法)是稳定、可靠的。如果把代码实现写到接口中,那接口就绑定了可能变化的因素,这就会导致实现不再稳定和可靠,是随时都可能被抛弃,被更改,被重构的。所以,接口中虽然可以有实现,但应该避免实现。
三十二、静态变量一定要先声明后赋值
静态变量是在类初始化时首先被加载的,JVM会去查找类中所有的静态声明,然后分配空间,注意这个时候只是分配了空间,还没有赋值,之后JVM会根据类中静态赋值(包括静态类赋值和静态块赋值)的先后顺序来执行。
有些程序员喜欢把变量定义放到类的底部,如果这是实例变量还好说,么有任何问题。但如果是静态变量,而且还在静态块中进行了赋值,那结果可就和你期望的不一样了。所以遵循Java通用的开发规范“变量先声明后使用” 是一个良好的编码风格。
三十三、不要覆写静态方法
一个实例对象有两个类型:表面类型和实际类型,表面类型是声明时的类型,实际类型是对象产生时的类型。对于非静态方法,它是根据对相关的实际类型来执行的,而对于静态方法来说就比较特殊了,首先静态方法不依赖实例对象,它是通过类名访问的;其次可以通过对象访问静态方法,如果是通过对象调用静态方法,JVM则会通过对象的表面类型查找到静态方法的入口,继而执行之。
在子类中构建与父类相同的方法名、输入参数、输出参数、访问权限(权限可以扩大),并且父类、子类都是静态方法,此种行为叫做“隐藏”。它与覆写有两点不同:
- 表现形式不同。隐藏用于静态方法,覆写用于非静态方法。
- 职责不同。隐藏的目的是为了抛弃父类静态方法,重现子类方法。
三十四、 构造函数尽量简化
构造函数简化,再简化,应该达到“一眼洞穿”的境界。
三十五、避免在构造函数中初始化其他类
不要在构造函数中声明初始化其他类,养成良好的习惯。
三十六、使用构造代码块精炼程序
构造代码块可以应用到如下场景:
- 初始化实例变量
- 初始化实例环境
以上两个场景利用了构造代码块的两个特性:在每个构造函数都运行和在构造函数中它首先运行。
三十七、构造代码块会想你所想
有一种特殊的情况:如果遇到this关键字(构造函数调用自身其他的构造函数时)则不插入构造代码块。在构造代码块的处理上,super方法没有任何特殊的地方,编译器只是把构造代码块插入到super关键字之后执行而已,仅此不同。
三十八、使用静态内部类提高封装性
静态内部类有两个优点:
- 加强了类的封装性
- 提高了代码的可读性
外部类和静态内部类不是组合关系,而是有强关联关系。
静态内部类和普通内部类的区别: - 静态内部类不持有外部类的引用。普通类可以访问外部类的属性和方法。静态内部类只能访问外部类的static属性和static方法。
- 静态内部类不依赖外部类。普通内部类和外部类之间是相互依赖关系,内部类实例不能脱离外部类实例,也就是说他们同生死,一起声明,一起被垃圾回收器回收。而静态内部类是独立存在的,即使外部类消亡了,静态内部类还是可以存在的。
- 普通内部类不能声明static变量(final static除外)和static方法
三十九、使用匿名类的构造函数
匿名类可以用一个初始化块充当构造函数。