CS-Notes/notes/面向对象思想.md

340 lines
12 KiB
Markdown
Raw Normal View History

2018-03-12 12:22:21 +08:00
<!-- GFM-TOC -->
* [S.O.L.I.D](#solid)
* [1. 单一责任原则](#1-单一责任原则)
* [2. 开放封闭原则](#2-开放封闭原则)
* [3. 里氏替换原则](#3-里氏替换原则)
* [4. 接口分离原则](#4-接口分离原则)
* [5. 依赖倒置原则](#5-依赖倒置原则)
* [其他常见原则](#其他常见原则)
* [1. 迪米特法则](#1-迪米特法则)
* [2. 合成复用原则](#2-合成复用原则)
* [3. 共同封闭原则](#3-共同封闭原则)
* [4. 稳定抽象原则](#4-稳定抽象原则)
* [5. 稳定依赖原则](#5-稳定依赖原则)
* [封装、继承、多态](#封装继承多态)
* [1. 封装](#1-封装)
* [2. 继承](#2-继承)
* [3. 多态](#3-多态)
* [UML](#uml)
* [1. 类图](#1-类图)
* [2. 时序图](#2-时序图)
* [参考资料](#参考资料)
<!-- GFM-TOC -->
# S.O.L.I.D
S.O.L.I.D 是面向对象设计和编程 (OOD&OOP) 中几个重要编码原则 (Programming Priciple) 的首字母缩写。
| 简写 | 全拼 | 中文翻译 |
| -- | -- | -- |
|SRP| The Single Responsibility Principle | 单一责任原则 |
|OCP| The Open Closed Principle | 开放封闭原则 |
|LSP| The Liskov Substitution Principle | 里氏替换原则 |
|ISP| The Interface Segregation Principle | 接口分离原则 |
|DIP| The Dependency Inversion Principle | 依赖倒置原则 |
## 1. 单一责任原则
2018-03-02 10:58:04 +08:00
当需要修改某个类的时候原因有且只有一个。换句话说就是让一个类只做一种类型责任,当这个类需要承当其他类型的责任的时候,就需要分解这个类。
2018-03-12 12:22:21 +08:00
## 2. 开放封闭原则
2018-03-02 10:58:04 +08:00
软件实体应该是可扩展,而不可修改的。也就是说,对扩展是开放的,而对修改是封闭的。
2018-03-12 12:22:21 +08:00
## 3. 里氏替换原则
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
当一个子类的实例应该能够替换任何其超类的实例时,它们之间才具有 is-a 关系。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
## 4. 接口分离原则
2018-03-02 10:58:04 +08:00
不能强迫用户去依赖那些他们不使用的接口。换句话说,使用多个专门的接口比使用单一的总接口总要好。
2018-03-12 12:22:21 +08:00
## 5. 依赖倒置原则
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
1. 高层模块不应该依赖于低层模块,二者都应该依赖于抽象
2. 抽象不应该依赖于细节,细节应该依赖于抽象
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
# 其他常见原则
2018-03-11 17:09:04 +08:00
除了上述的经典原则,在实际开发中还有下面这些常见的设计原则。
2018-03-12 12:22:21 +08:00
| 简写 | 全拼 | 中文翻译 |
| -- | -- | -- |
|LoD| The Law of Demeter | 迪米特法则 |
|CRP| The Composite Reuse Principle | 合成复用原则 |
|CCP| The Common Closure Principle | 共同封闭原则 |
|SAP| The Stable Abstractions Principle | 稳定抽象原则 |
|SDP| The Stable Dependencies Principle | 稳定依赖原则 |
2018-03-11 17:09:04 +08:00
2018-03-12 12:22:21 +08:00
## 1. 迪米特法则
2018-03-11 17:09:04 +08:00
2018-03-12 12:22:21 +08:00
迪米特法则又叫作最少知道原则Least Knowledge Principle 简写LKP就是说一个对象应当对其他对象有尽可能少的了解不和陌生人说话。
2018-03-11 17:09:04 +08:00
2018-03-12 12:22:21 +08:00
## 2. 合成复用原则
2018-03-11 17:09:04 +08:00
尽量使用对象组合,而不是继承来达到复用的目的。
2018-03-12 12:22:21 +08:00
## 3. 共同封闭原则
2018-03-11 17:09:04 +08:00
一起修改的类,应该组合在一起(同一个包里)。如果必须修改应用程序里的代码,我们希望所有的修改都发生在一个包里(修改关闭),而不是遍布在很多包里。
2018-03-12 12:22:21 +08:00
## 4. 稳定抽象原则
2018-03-11 17:09:04 +08:00
最稳定的包应该是最抽象的包,不稳定的包应该是具体的包,即包的抽象程度跟它的稳定性成正比。
2018-03-12 12:22:21 +08:00
## 5. 稳定依赖原则
2018-03-11 17:09:04 +08:00
包之间的依赖关系都应该是稳定方向依赖的,包要依赖的包要比自己更具有稳定性。
2018-03-12 12:22:21 +08:00
# 封装、继承、多态
2018-03-02 10:58:04 +08:00
封装、继承、多态是面向对象的三大特性。
2018-03-12 12:22:21 +08:00
## 1. 封装
2018-03-02 10:58:04 +08:00
利用抽象数据类型将数据和基于数据的操作封装在一起,使其构成一个不可分割的独立实体,数据被保护在抽象数据类型的内部,尽可能地隐藏内部的细节,只保留一些对外接口使之与外部发生联系。用户是无需知道对象内部的细节,但可以通过该对象对外的提供的接口来访问该对象。
封装有三大好处:
2018-03-12 12:22:21 +08:00
1. 良好的封装能够减少耦合。
2. 类内部的结构可以自由修改。
3. 可以对成员进行更精确的控制。
4. 隐藏信息,实现细节。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
以下 Person 类封装 name、gender、age 等属性,外界只能通过 get() 方法获取一个 Person 对象的 name 属性和 gender 属性,而无法获取 age 属性,但是 age 属性可以供 work() 方法使用。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
注意到 gender 属性使用 int 数据类型进行存储,封装使得用户注意不到这种实现细节。并且在需要修改使用的数据类型时,也可以在不影响客户端代码的情况下进行。
2018-03-02 10:58:04 +08:00
```java
2018-03-12 12:22:21 +08:00
public class Person {
private String name;
private int gender;
private int age;
public String getName() {
return name;
}
public String getGender() {
return gender == 0 ? "man" : "woman";
}
public void work() {
if(18 <= age && age <= 50) {
System.out.println(name + " is working very hard!");
} else {
System.out.println(name + " can't work!");
}
}
2018-03-02 10:58:04 +08:00
}
```
2018-03-12 12:22:21 +08:00
## 2. 继承
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
继承实现了 **is-a** 关系,例如 Cat 和 Animal 就是一种 is-a 关系,因此可以将 Cat 继承自 Animal从而获得 Animal 非 private 的属性和方法。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
Cat 可以当做 Animal 来使用,也就是可以使用 Animal 引用 Cat 对象,这种子类转换为父类称为 **向上转型**
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
继承应该遵循里氏替换原则:当一个子类的实例应该能够替换任何其超类的实例时,它们之间才具有 is-a 关系。
2018-03-02 10:58:04 +08:00
```java
2018-03-12 12:22:21 +08:00
Animal animal = new Cat();
2018-03-02 10:58:04 +08:00
```
2018-03-12 12:22:21 +08:00
## 3. 多态
2018-03-02 10:58:04 +08:00
多态分为编译时多态和运行时多态。编译时多态主要指方法的重装,运行时多态指程序中定义的对象引用所指向的具体类型在运行期间才确定。
2018-03-12 12:22:21 +08:00
多态有三个条件1. 继承2. 覆盖父类方法3. 向上转型。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
下面的代码中乐器类Instrument有两个子类Wind 和 Percussion它们都覆盖了 play() 方法,并且在 main() 方法中使用父类 Instrument 来引用 Wind 和 Percussion 对象。在 Instrument 引用调用 play() 方法时,会执行实际引用对象所在类的 play() 方法,而不是 Instrument 类的方法。
2018-03-02 10:58:04 +08:00
```java
2018-03-12 12:22:21 +08:00
public class Instrument {
public void play() {
System.out.println("Instument is playing...");
}
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
public class Wind extends Instrument {
public void play() {
System.out.println("Wind is playing...");
}
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
public class Percussion extends Instrument {
public void play() {
System.out.println("Percussion is playing...");
}
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
public class Music {
public static void main(String[] args) {
List<Instrument> instruments = new ArrayList<>();
instruments.add(new Wind());
instruments.add(new Percussion());
for(Instrument instrument : instruments) {
instrument.play();
}
}
2018-03-02 10:58:04 +08:00
}
```
2018-03-12 12:22:21 +08:00
# UML
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
## 1. 类图
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
**1.1 继承相关**
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
继承有两种形式 : 泛化generalize和实现realize表现为 is-a 关系。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
① 泛化关系 (generalization)
2018-03-02 10:58:04 +08:00
从具体类中继承
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//29badd92-109f-4f29-abb9-9857f5973928.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
② 实现关系 (realize)
2018-03-02 10:58:04 +08:00
从抽象类或者接口中继承
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//4b16e1d3-3a60-472c-9756-2f31b1c48abe.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
**1.2 整体和部分**
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
① 聚合关系 (aggregation)
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
表示整体由部分组成,但是整体和部分不是强依赖的,整体不存在了部分还是会存在。以下表示 B 由 A 组成:
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//34259bb8-ca3a-4872-8771-9e946782d9c3.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
② 组合关系 (composition)
2018-03-02 10:58:04 +08:00
和聚合不同,组合中整体和部分是强依赖的,整体不存在了部分也不存在了。比如公司和部门,公司没了部门就不存在了。但是公司和员工就属于聚合关系了,因为公司没了员工还在。
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//7dda050d-ac35-4f47-9f51-18f18ed6fa9a.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
**1.3 相互联系**
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
① 关联关系 (association)
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
表示不同类对象之间有关联,这是一种静态关系,与运行过程的状态无关,在最开始就可以确定。因此也可以用 1 对 1、多对 1、多对多这种关联关系来表示。比如学生和学校就是一种关联关系一个学校可以有很多学生但是一个学生只属于一个学校因此这是一种多对一的关系在运行开始之前就可以确定。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//4ccd294c-d6b2-421b-839e-d88336ff5fb7.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
② 依赖关系 (dependency)
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
和关联关系不同的是 , 依赖关系是在运行过程中起作用的。一般依赖作为类的构造器或者方法的参数传入。双向依赖时一种不好的设计。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//47ca2614-509f-476e-98fc-50ec9f9d43c0.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
## 2. 时序图
2018-03-02 10:58:04 +08:00
2018-03-03 15:49:44 +08:00
http://www.cnblogs.com/wolf-sun/p/UML-Sequence-diagram.html
2018-03-12 12:22:21 +08:00
**2.1 定义**
2018-03-02 10:58:04 +08:00
时序图描述了对象之间传递消息的时间顺序,它用来表示用例的行为顺序。它的主要作用是通过对象间的交互来描述用例(注意是对象),从而寻找类的操作。
2018-03-12 12:22:21 +08:00
**2.2 赤壁之战时序图**
2018-03-02 10:58:04 +08:00
从虚线从上往下表示时间的推进。
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//80c5aff8-fc46-4810-aeaa-215b5c60a003.png"/> </div><br>
2018-03-02 10:58:04 +08:00
可见,通过时序图可以知道每个类具有以下操作:
```java
2018-03-12 12:22:21 +08:00
publc class 刘备 {
public void 应战 ();
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
publc class 孔明 {
public void 拟定策略 ();
public void 联合孙权 ();
private void 借东风火攻 ();
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
public class 关羽 {
public void 防守荊州 ();
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
public class 张飞 {
public void 防守荆州前线 ();
2018-03-02 10:58:04 +08:00
}
2018-03-12 12:22:21 +08:00
public class 孙权 {
public void 领兵相助 ();
2018-03-02 10:58:04 +08:00
}
```
2018-03-12 12:22:21 +08:00
**2.3 活动图、时序图之间的关系**
2018-03-02 10:58:04 +08:00
活动图示从用户的角度来描述用例;
时序图是从计算机的角度(对象间的交互)描述用例。
2018-03-12 12:22:21 +08:00
**2.4 类图与时序图的关系**
2018-03-02 10:58:04 +08:00
类图描述系统的静态结构,时序图描述系统的动态行为。
2018-03-12 12:22:21 +08:00
**2.5 时序图的组成**
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
① 对象
2018-03-02 10:58:04 +08:00
有三种表现形式
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//25b8adad-2ef6-4f30-9012-c306b4e49897.png"/> </div><br>
2018-03-02 10:58:04 +08:00
在画图时,应该遵循以下原则:
2018-03-12 12:22:21 +08:00
1. 把交互频繁的对象尽可能地靠拢。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
2. 把初始化整个交互活动的对象(有时是一个参与者)放置在最左边。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
② 生命线
2018-03-02 10:58:04 +08:00
生命线从对象的创建开始到对象销毁时终止
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//b7b0eac6-e7ea-4fb6-8bfb-95fec6f235e2.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
③ 消息
2018-03-02 10:58:04 +08:00
对象之间的交互式通过发送消息来实现的。
2018-03-12 12:22:21 +08:00
消息有 4 种类型:
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
1\. 简单消息,不区分同步异步。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//a13b62da-0fa8-4224-a615-4cadacc08871.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
2\. 同步消息,发送消息之后需要暂停活动来等待回应。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//33821037-dc40-4266-901c-e5b38e618426.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
3\. 异步消息,发送消息之后不需要等待。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//dec6c6cc-1b5f-44ed-b8fd-464fcf849dac.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
4\. 返回消息,可选。
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
④ 激活
2018-03-02 10:58:04 +08:00
生命线上的方框表示激活状态,其它时间处于休眠状态。
2018-03-12 12:22:21 +08:00
<div align="center"> <img src="../pics//6ab5de9b-1c1e-4118-b2c3-fb6c7ed7de6f.png"/> </div><br>
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
# 参考资料
2018-03-02 10:58:04 +08:00
2018-03-12 12:22:21 +08:00
- Java 编程思想
- [ 面向对象设计的 SOLID 原则 ](http://www.cnblogs.com/shanyou/archive/2009/09/21/1570716.html)
- [ 看懂 UML 类图和时序图 ](http://design-patterns.readthedocs.io/zh_CN/latest/read_uml.html#generalization)
- [UML 系列——时序图顺序图sequence diagram](http://www.cnblogs.com/wolf-sun/p/UML-Sequence-diagram.html)
- [ 面向对象编程三大特性 ------ 封装、继承、多态 ](http://blog.csdn.net/jianyuerensheng/article/details/51602015)