1. 概念和特点
类的基本作用就是封装代码。封装将类的一些特征和行为隐藏在类内部,不允许类外部直接访问。
封装可以被认为是一个保护屏障,防止该类的代码和数据被外部类定义的代码随机访问。
我们可以通过类提供的方法来实现对隐藏信息的操作和访问。隐藏了对象的信息,留出了访问的接口。
在我们日常生活中,封装与我们息息相关,智能手机就是一个拥有良好封装的例子,我们不需要关心其内部复杂的逻辑电路设计,可以通过手机的屏幕、按键、充电口、耳机接口等等外部接口来对手机进行操作和使用。复杂的逻辑电路以及模块被封装在手机的内部,而留出的这些必要接口,让我们更加简便地使用手机的同时也保护了手机的内部细节。
封装有两个特点:
- 只能通过规定的方法访问数据;
- 隐藏类的实例细节,方便修改和实现。
2. 为什么需要封装
封装具有以下优点:
- 封装有利于提高类的内聚性,适当的封装可以让代码更容易理解与维护;
- 良好的封装有利于降低代码的耦合度;
- 一些关键属性只允许类内部可以访问和修改,增强类的安全性;
- 隐藏实现细节,为调用方提供易于理解的接口;
- 当需求发生变动时,我们只需要修改我们封装的代码,而不需要到处修改调用处的代码。
3. 实现封装
在 Java 语言中,如何实现封装呢?需要 3 个步骤。
- 修改属性的可见性为
private
; - 创建公开的 getter 和 setter 方法,分别用于属性的读写;
- 在 getter 和 setter 方法中,对属性的合法性进行判断。
我们来看一个 NBA 球员类NBAPlayer
:
- 1
- 2
- 3
- 4
- 5
- 6
在类内部(即类名后面{}
之间的区域)定义了成员属性name
和age
,我们知道,在类外部调用处可以对其属性进行修改:
- 1
- 2
如下是实例代码:
运行结果:
- 1
我们通过对象名.属性名的方式对age
赋值为 -1
,显然,球员的年龄为-1
是反常理的。
下面我们对NBAPlayer
类进行封装。
- 我们可以使用私有化访问控制符修饰类内部的属性,让其只在类的内部可以访问:
- 1
- 2
- 3
private
关键字限定了其修饰的成员只能在类内部访问,这样之后就无法在类外部使用player.age =-1
这样的赋值方式进行赋值了。
- 创建公开的(public)
getter
和setter
方法:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
顾名思义,getter
就是取属性值,setter
就是给属性赋值,这样在类的外部就可以通过调用其方法对属性进行操作了。
- 对属性进行逻辑判断,以
age
属性的setter
方法为例:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
在setAge
方法中,我们将参数age
小于 0 的情况进行了处理,如果小于 0,直接将age
赋值为0。除了给默认值的方式,我们也可以抛出异常,提示调用方传参不合法。
在类外部对属性进行读写:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
试想,如果在类外部,有很多地方都会操作属性值,当属性值读写逻辑发生改变时,我们只需修改类内部的逻辑。
另外,对于有参构造方法中,对属性赋值时,直接调用其setter
方法。无需再写重复的逻辑判断,提高代码复用性:
- 1
- 2
- 3
如下是实现封装后完整实例代码:
运行结果:
- 1
- 2
- 3
- 4
- 5
4. 小结
面向对象的三大特征:封装、继承、多态。
封装隐藏了对象的信息,并且留出了访问的接口。