《第4章 类和接口》
类和接口是Java语言的核心,本章包含的一些指导原则,可以帮助你更好地设计出更加有用、健壮、灵活的类和接口。
【第12条】使类和成员的可访问能力最小化
要想区别一个设计良好的模块与一个设计不好的模块,最重要的因素是,这个模块对于外部的其它模块而言,是否隐藏了内部的数据和其他的实现细节。换句话说,就是模块的设计者是否对其进行了良好的封装。
经验表明,你应该尽可能地使每一个类或成员不被外界访问。也就是说,在保证与该模块相关的程序能够正确运行的前提下,尽可能使用最低可能的访问级别。那些一上来不加思索就统统 public 的做法显然是要着重声讨的。
对于顶层的(非嵌套的)类和接口,它们只有2种访问级别:包级私有(package-private)和公有(public)。如果选择了包级私有,那么它只是这个包的实现的一部分,而不是该包对外提供服务的API的一部分。在以后的版本中,你可以对它进行修改、替换甚至删除,而无需担心会伤害到现有的使用者。而如果选择的公有的,你就有义务永远支持它,以保持兼容性。
对于成员(域和方法),访问级别共为4种:
● 私有的(private)——只有该成员的顶层类中才能访问
● 包级私有(package-private)——本包内任何类都可访问
● 保护的(protected)——本包内的任何类和所在类的子类都可以访问
● 公有的(public)——任何地方都可以访问
私有、保护、公有 是几乎所有人都知道的,但是包级私有取不敢这么说。再有一点就是保护的,一来说“所在类的子类都可以访问”是尽人皆知的,但和包级私有一样“本包内任何类都可访问”这一点同样不敢说是尽人皆知。
也正是这个“包级私有”得以让我们既可以按功能分解为不同的类,又可以不对外部开发访问权限。其实就我个人来说,如果能有一个“反import”,用以写明只对谁谁谁可访问才好呢,这样就不必把这样的一些类都拘泥于同一个包内了。
那么“包级私有”的修饰符是什么呢?其实什么都不写就是包级私有,如:
class PackagePrivateClass {
String name;
.....
}
看到这样的代码,如果是来自“高手”的,你大可不必去担心“这家伙是不是偷懒,或者疏忽漏掉了修饰符?”。但是,如果这段代码是来自“基层”的,我还是建议你最好多问个问什么。
最后,有一个小知识,或者叫小结论:具有公有的静态final数组域几乎总是错误的。注意这句话共有4个定语——公有的、静态的、final的、数组。
// 这可以肯定是错误的
public static final Type[] VALUES = { ... };
// 应该改成这样
private static final Type[] PRIVATE_VALUES = { ... };
public static final List VALUE = Collections.unmodifiableList(Arrays.asList(PRIVATE_VALUES));
// 或者这样写(可能损失一点性能),前提是Type实现了克隆接口,而且你不反感克隆
private static final Type[] PRIVATE_VALUES = { ... };
public static final Type[] values() {
return (Type[]) PRIVATE_VALUES.clone();
}
总结一下,你应该总是尽量降低可访问性,否则你就有义务对你开放出去的东西负责到底。
看完这一条,我要做的是,重新审视一下我的 xxxx.util.support 包,这个包中的类都是用来对 util 里的类做支持的(幕后英雄)。根据本条的“精神”,我取消了 util.support包,转而把这些支持类都放入 util 包里,做成了包级私有类,这样这些support类就仅仅对util中的类负责了,在外部就看不到它们了,我就没有义务使它们永远保持兼容了。
另外,还有一点就是在util包中找到那些仅仅是为Framework中某些类服务的工具类,这些类的设计目的就不是给Framework的使用者用的,所以同样地把它们转移到被调用者的包里,做成包级私有(好在没有同时对两个及以上包提供服务的util类)。util包中只保留那些目的是给外部使用者服务的类。
【Effective Java 学习笔记】系列连载专题请见:
http://tonylian.iteye.com/categories/64208
分享到:
相关推荐
在VB.NET编程中,"最小化代码"通常指的是通过各种技术手段减少代码的体积,提高程序的可读性、可维护性和执行效率。这涉及到代码优化、重构、使用内置函数和语言特性等多个方面。以下是一些关于如何在VB.NET中实现...
14. **最小化重定向**:每增加一次重定向,就增加了额外的请求时间和等待时间,应尽量避免不必要的重定向。 15. **移除重复的JavaScript**:确保没有重复加载同一脚本,可以避免不必要的下载和执行时间。 16. **...
- **1.6.7.2 属性**:提供了对私有字段的访问和修改能力。 - **1.6.7.3 索引器**:类似于数组下标的方式访问对象的成员。 - **1.6.7.4 事件**:允许对象之间进行通信。 - **1.6.7.5 运算符**:自定义运算符以...
16. 当希望一个类中的成员不能在类的外部访问时,应使用private修饰符定义该成员。 17. 使用public修饰符定义的成员可以在类外部通过对象来直接访问。 五、Java类和对象 18. 使用final修饰符定义的变量,使得该...
第 7章到第12 章 第三篇 集中在 C++为基于过程化的程序设计所提供的支持上 第 7 章介绍C++函数机制 函数封装了一组操作 它们通常形成一项单一的任务 如 print() 名 字后面的括号表明它是一个函数 关于程序域和...
#### 第十二部分:JFC(Java Foundation Classes) - 包含Swing组件库,提供更丰富的GUI功能。 #### 第十三部分:Applets - 过时的技术,曾用于编写嵌入网页中的小型应用程序。 #### 第十四部分:线程 - 线程是...
解释:这是一道典型的项目调度问题,需要通过合理安排小明和家人的过桥顺序来最小化总时间。 知识点:项目调度、流程优化、资源分配。 这份笔试题目涵盖了Java语言、操作系统、计算机网络、算法设计等多个领域的...
14. **友员函数**:友员函数不是类的成员,但可以访问类的私有和保护成员。一个友员函数可以是多个类的友员,可以在类的外部定义,并且通过`friend`关键字声明。 15. **友员的作用**:友员函数可以提高程序的运行...
`public`成员可以被任何地方访问,`protected`成员在类和其派生类中可见,而`private`成员只在类内部可见。掌握这些访问控制符的使用对于实现封装和保护类的内部状态至关重要。 ### 28. const限定符的使用 `const`...
**3.5 成员访问** 成员可以通过不同的方式访问,包括公共、私有等。 - **已声明可访问性**:定义了成员的可见性级别。 - **可访问域**:指定了成员可以被访问的范围。 - **实例成员的受保护访问**:允许派生类访问...
面向对象方法所强调的基本原则,就是直接面对客观存在的事物来进行软件开发,将人们在日常生活中习惯的思维方式和表达方式应用在软件开发中,使软件开发从过分专业化的方法、规则和技巧中回到客观世界,回到人们通常...
22. **最小化协作者**:减少类与其协作者的数量,降低系统的复杂度。 23. **减少消息传递**:减少类与其协作者之间传递的消息数量,简化通信路径。 24. **减少协作量**:减少类与其协作者之间的不同消息数量,优化...
条款13: 初始化列表中成员列出的顺序和它们在类中声明的顺序相同 条款14: 确定基类有虚析构函数 条款15: 让operator=返回*this的引用 条款16: 在operator=中对所有数据成员赋值 条款17: 在operator=中检查给自己赋值...
第8章 类和对象 8.1 面向对象程序设计方法概述 8.1.1 什么是面向对象的程序设计 8.1.2 面向对象程序设计的特点 8.1.3 类和对象的作用 8.1.4 面向对象的软件开发 8.2 类的声明和对象的定义 8.2.1 类和对象的关系 ...
5. **访问修饰符**:public、private、protected和默认访问权限在控制类和成员的可见性方面的作用,有助于理解封装原则。 6. **接口与抽象类**:接口定义了完全抽象的行为规范,而抽象类可以包含实现部分。两者都是...