计算机操作系统.md 45.4 KB
Newer Older
C
CyC2018 已提交
1 2 3 4 5 6 7 8 9 10
<!-- GFM-TOC -->
* [一、概述](#一概述)
    * [操作系统基本特征](#操作系统基本特征)
    * [操作系统基本功能](#操作系统基本功能)
    * [系统调用](#系统调用)
    * [大内核和微内核](#大内核和微内核)
    * [中断分类](#中断分类)
* [二、进程管理](#二进程管理)
    * [进程与线程](#进程与线程)
    * [进程状态的切换](#进程状态的切换)
C
CyC2018 已提交
11
    * [进程调度算法](#进程调度算法)
C
CyC2018 已提交
12
    * [进程调度算法实现](#进程调度算法实现)
C
CyC2018 已提交
13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
    * [进程同步](#进程同步)
    * [经典同步问题](#经典同步问题)
    * [进程通信](#进程通信)
* [三、死锁](#三死锁)
    * [死锁的必要条件](#死锁的必要条件)
    * [死锁的处理方法](#死锁的处理方法)
* [四、内存管理](#四内存管理)
    * [虚拟内存](#虚拟内存)
    * [分页系统地址映射](#分页系统地址映射)
    * [页面置换算法](#页面置换算法)
    * [分段](#分段)
    * [段页式](#段页式)
    * [分页与分段的比较](#分页与分段的比较)
* [五、设备管理](#五设备管理)
    * [磁盘调度算法](#磁盘调度算法)
* [六、链接](#六链接)
    * [编译系统](#编译系统)
    * [静态链接](#静态链接)
C
CyC2018 已提交
31
    * [目标文件](#目标文件)
C
CyC2018 已提交
32 33 34 35 36 37 38 39 40 41
    * [动态链接](#动态链接)
* [参考资料](#参考资料)
<!-- GFM-TOC -->


# 一、概述

## 操作系统基本特征

### 1. 并发
C
CyC2018 已提交
42

C
CyC2018 已提交
43
并发是指宏观上在一段时间内能同时运行多个程序,而并行则指同一时刻能运行多个指令。
C
CyC2018 已提交
44 45 46 47 48

并行需要硬件支持,如多流水线或者多处理器。

操作系统通过引入进程和线程,使得程序能够并发运行。

C
CyC2018 已提交
49
### 2. 共享
C
CyC2018 已提交
50

C
CyC2018 已提交
51
共享是指系统中的资源可以被多个并发进程共同使用。
C
CyC2018 已提交
52 53 54 55 56

有两种共享方式:互斥共享和同时共享。

互斥共享的资源称为临界资源,例如打印机等,在同一时间只允许一个进程访问,需要用同步机制来实现对临界资源的访问。

C
CyC2018 已提交
57
### 3. 虚拟
C
CyC2018 已提交
58 59 60 61 62

虚拟技术把一个物理实体转换为多个逻辑实体。

主要有两种虚拟技术:时分复用技术和空分复用技术。例如多个进程能在同一个处理器上并发执行使用了时分复用技术,让每个进程轮流占有处理器,每次只执行一小个时间片并快速切换。

C
CyC2018 已提交
63
### 4. 异步
C
CyC2018 已提交
64 65 66

异步指进程不是一次性执行完毕,而是走走停停,以不可知的速度向前推进。

C
CyC2018 已提交
67
## 操作系统基本功能
C
CyC2018 已提交
68

C
CyC2018 已提交
69
### 1. 进程管理
C
CyC2018 已提交
70 71 72

进程控制、进程同步、进程通信、死锁处理、处理机调度等。

C
CyC2018 已提交
73
### 2. 内存管理
C
CyC2018 已提交
74

C
CyC2018 已提交
75
内存分配、地址映射、内存保护与共享、虚拟内存等。
C
CyC2018 已提交
76

C
CyC2018 已提交
77
### 3. 文件管理
C
CyC2018 已提交
78

C
CyC2018 已提交
79
文件存储空间的管理、目录管理、文件读写管理和保护等。
C
CyC2018 已提交
80

C
CyC2018 已提交
81
### 4. 设备管理
C
CyC2018 已提交
82

C
CyC2018 已提交
83
完成用户的 I/O 请求,方便用户使用各种设备,并提高设备的利用率。
C
CyC2018 已提交
84 85

主要包括缓冲管理、设备分配、设备处理、虛拟设备等。
C
CyC2018 已提交
86

C
CyC2018 已提交
87
## 系统调用
C
CyC2018 已提交
88 89 90

如果一个进程在用户态需要使用内核态的功能,就进行系统调用从而陷入内核,由操作系统代为完成。

C
CyC2018 已提交
91
<div align="center"> <img src="../pics//tGPV0.png" width="600"/> </div><br>
C
CyC2018 已提交
92

C
CyC2018 已提交
93
Linux 的系统调用主要有以下这些:
C
CyC2018 已提交
94

C
CyC2018 已提交
95 96 97 98 99 100 101 102
| Task | Commands |
| :---: | --- |
| 进程控制 | fork(); exit(); wait(); |
| 进程通信 | pipe(); shmget(); mmap(); |
| 文件操作 | open(); read(); write(); |
| 设备操作 | ioctl(); read(); write(); |
| 信息维护 | getpid(); alarm(); sleep(); |
| 安全 | chmod(); umask(); chown(); |
C
CyC2018 已提交
103

C
CyC2018 已提交
104
## 大内核和微内核
C
CyC2018 已提交
105

C
CyC2018 已提交
106
### 1. 大内核
C
CyC2018 已提交
107 108 109 110 111

大内核是将操作系统功能作为一个紧密结合的整体放到内核。

由于各模块共享信息,因此有很高的性能。

C
CyC2018 已提交
112
### 2. 微内核
C
CyC2018 已提交
113 114 115 116 117 118 119

由于操作系统不断复杂,因此将一部分操作系统功能移出内核,从而降低内核的复杂性。移出的部分根据分层的原则划分成若干服务,相互独立。

在微内核结构下,操作系统被划分成小的、定义良好的模块,只有微内核这一个模块运行在内核态,其余模块运行在用户态。

因为需要频繁地在用户态和核心态之间进行切换,所以会有一定的性能损失。

C
CyC2018 已提交
120
<div align="center"> <img src="../pics//2_14_microkernelArchitecture.jpg"/> </div><br>
C
CyC2018 已提交
121

C
CyC2018 已提交
122
## 中断分类
C
CyC2018 已提交
123

C
CyC2018 已提交
124
### 1. 外中断
C
CyC2018 已提交
125

C
CyC2018 已提交
126
由 CPU 执行指令以外的事件引起,如 I/O 完成中断,表示设备输入/输出处理已经完成,处理器能够发送下一个输入/输出请求。此外还有时钟中断、控制台中断等。
C
CyC2018 已提交
127

C
CyC2018 已提交
128
### 2. 异常
C
CyC2018 已提交
129

C
CyC2018 已提交
130
由 CPU 执行指令的内部事件引起,如非法操作码、地址越界、算术溢出等。
C
CyC2018 已提交
131

C
CyC2018 已提交
132
### 3. 陷入
C
CyC2018 已提交
133 134 135

在用户程序中使用系统调用。

C
CyC2018 已提交
136
# 二、进程管理
C
CyC2018 已提交
137

C
CyC2018 已提交
138
## 进程与线程
C
CyC2018 已提交
139

C
CyC2018 已提交
140
### 1. 进程
C
CyC2018 已提交
141 142 143

进程是资源分配的基本单位。

C
CyC2018 已提交
144
进程控制块 (Process Control Block, PCB) 描述进程的基本信息和运行状态,所谓的创建进程和撤销进程,都是指对 PCB 的操作。
C
CyC2018 已提交
145

C
CyC2018 已提交
146
下图显示了 4 个程序创建了 4 个进程,这 4 个进程可以并发地执行。
C
CyC2018 已提交
147

C
CyC2018 已提交
148
<div align="center"> <img src="../pics//a6ac2b08-3861-4e85-baa8-382287bfee9f.png"/> </div><br>
C
CyC2018 已提交
149

C
CyC2018 已提交
150
### 2. 线程
C
CyC2018 已提交
151 152 153 154 155

线程是独立调度的基本单位。

一个进程中可以有多个线程,它们共享进程资源。

C
CyC2018 已提交
156
<div align="center"> <img src="../pics//3cd630ea-017c-488d-ad1d-732b4efeddf5.png"/> </div><br>
C
CyC2018 已提交
157

C
CyC2018 已提交
158
### 3. 区别
C
CyC2018 已提交
159

C
CyC2018 已提交
160
- 拥有资源:进程是资源分配的基本单位,但是线程不拥有资源,线程可以访问隶属进程的资源。
C
CyC2018 已提交
161

C
CyC2018 已提交
162
- 调度:线程是独立调度的基本单位,在同一进程中,线程的切换不会引起进程切换,从一个进程内的线程切换到另一个进程中的线程时,会引起进程切换。
C
CyC2018 已提交
163

C
CyC2018 已提交
164
- 系统开销:由于创建或撤销进程时,系统都要为之分配或回收资源,如内存空间、I/O 设备等,所付出的开销远大于创建或撤销线程时的开销。类似地,在进行进程切换时,涉及当前执行进程 CPU 环境的保存及新调度进程 CPU 环境的设置,而线程切换时只需保存和设置少量寄存器内容,开销很小。
C
CyC2018 已提交
165

C
CyC2018 已提交
166
- 通信方面:进程间通信 (IPC) 需要进程同步和互斥手段的辅助,以保证数据的一致性。而线程间可以通过直接读/写同一进程中的数据段(如全局变量)来进行通信。
C
CyC2018 已提交
167

C
CyC2018 已提交
168
举例:QQ 和浏览器是两个进程,浏览器进程里面有很多线程,例如 HTTP 请求线程、事件响应线程、渲染线程等等,线程的并发执行使得在浏览器中点击一个新链接从而发起 HTTP 请求时,浏览器还可以响应用户的其它事件。
C
CyC2018 已提交
169

C
CyC2018 已提交
170
## 进程状态的切换
C
CyC2018 已提交
171

C
CyC2018 已提交
172
<div align="center"> <img src="../pics//ProcessState.png" width="500"/> </div><br>
C
CyC2018 已提交
173

C
CyC2018 已提交
174 175 176
- 就绪状态(ready):等待被调度
- 运行状态(running)
- 阻塞状态(waiting):等待资源
C
CyC2018 已提交
177 178 179

应该注意以下内容:

C
CyC2018 已提交
180 181
- 只有就绪态和运行态可以相互转换,其它的都是单向转换。就绪状态的进程通过调度算法从而获得 CPU 时间,转为运行状态;而运行状态的进程,在分配给它的 CPU 时间片用完之后就会转为就绪状态,等待下一次调度。
- 阻塞状态是缺少需要的资源从而由运行状态转换而来,但是该资源不包括 CPU 时间,缺少 CPU 时间会从运行态转换为就绪态。
C
CyC2018 已提交
182

C
CyC2018 已提交
183
## 进程调度算法
C
CyC2018 已提交
184

C
CyC2018 已提交
185
不同环境的调度算法目标不同,因此需要针对不同环境来讨论调度算法。
C
CyC2018 已提交
186

C
CyC2018 已提交
187
### 1. 批处理系统
C
CyC2018 已提交
188

C
CyC2018 已提交
189
批处理系统没有太多的用户操作,在该系统中,调度算法目标是保证吞吐量和周转时间(从提交到终止的时间)。
C
CyC2018 已提交
190

C
CyC2018 已提交
191 192 193
**1.1 先来先服务 first-come first-serverd(FCFS)** 

按照请求的顺序进行调度。
C
CyC2018 已提交
194 195 196

有利于长作业,但不利于短作业,因为短作业必须一直等待前面的长作业执行完毕才能执行,而长作业又需要执行很长时间,造成了短作业等待时间过长。

C
CyC2018 已提交
197
**1.2 短作业优先 shortest job first(SJF)** 
C
CyC2018 已提交
198

C
CyC2018 已提交
199
按估计运行时间最短的顺序进行调度。
C
CyC2018 已提交
200 201 202

长作业有可能会饿死,处于一直等待短作业执行完毕的状态。因为如果一直有短作业到来,那么长作业永远得不到调度。

C
CyC2018 已提交
203 204 205
**1.3 最短剩余时间优先 shortest remaining time next(SRTN)** 

按估计剩余时间最短的顺序进行调度。
C
CyC2018 已提交
206

C
CyC2018 已提交
207
### 2. 交互式系统
C
CyC2018 已提交
208

C
CyC2018 已提交
209
交互式系统有大量的用户交互操作,在该系统中调度算法的目标是快速地进行响应。
C
CyC2018 已提交
210

C
CyC2018 已提交
211
**2.1 时间片轮转** 
C
CyC2018 已提交
212

C
CyC2018 已提交
213
将所有就绪进程按 FCFS 的原则排成一个队列,每次调度时,把 CPU 时间分配给队首进程,该进程可以执行一个时间片。当时间片用完时,由计时器发出时钟中断,调度程序便停止该进程的执行,并将它送往就绪队列的末尾,同时继续把 CPU 时间分配给队首的进程。
C
CyC2018 已提交
214

C
CyC2018 已提交
215
时间片轮转算法的效率和时间片的大小有很大关系。因为进程切换都要保存进程的信息并且载入新进程的信息,如果时间片太小,会导致进程切换得太频繁,在进程切换上就会花过多时间。
C
CyC2018 已提交
216

C
CyC2018 已提交
217
<div align="center"> <img src="../pics//8c662999-c16c-481c-9f40-1fdba5bc9167.png"/> </div><br>
C
CyC2018 已提交
218

C
CyC2018 已提交
219
**2.2 优先级调度** 
C
CyC2018 已提交
220

C
CyC2018 已提交
221
为每个进程分配一个优先级,按优先级进行调度。
C
CyC2018 已提交
222

C
CyC2018 已提交
223
为了防止低优先级的进程永远等不到调度,可以随着时间的推移增加等待进程的优先级。
C
CyC2018 已提交
224

C
CyC2018 已提交
225
**2.3 多级反馈队列** 
C
CyC2018 已提交
226

C
CyC2018 已提交
227 228 229
如果一个进程需要执行 100 个时间片,如果采用时间片轮转调度算法,那么需要交换 100 次。

多级队列是为这种需要连续执行多个时间片的进程考虑,它设置了多个队列,每个队列时间片大小都不同,例如 1,2,4,8,..。进程在第一个队列没执行完,就会被移到下一个队列。这种方式下,之前的进程只需要交换 7 次。
C
CyC2018 已提交
230 231 232

每个队列优先权也不同,最上面的优先权最高。因此只有上一个队列没有进程在排队,才能调度当前队列上的进程。

C
CyC2018 已提交
233 234 235 236
可以将这种调度算法看成是时间片轮转调度算法和优先级调度算法的结合。

<div align="center"> <img src="../pics//042cf928-3c8e-4815-ae9c-f2780202c68f.png"/> </div><br>

C
CyC2018 已提交
237
### 3. 实时系统
C
CyC2018 已提交
238

C
CyC2018 已提交
239
实时系统要求一个请求在一个确定时间内得到响应。
C
CyC2018 已提交
240 241 242

分为硬实时和软实时,前者必须满足绝对的截止时间,后者可以容忍一定的超时。

C
CyC2018 已提交
243 244
## 进程调度算法实现

C
CyC2018 已提交
245
以下只是假象系统上的调度算法实现。源代码:[Scheduling](https://github.com/CyC2018/Algorithm/tree/master/Process-Scheduling/src)
C
CyC2018 已提交
246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540

### 1. FCFS

首先创建进程数据结构:

```java
public class Process {
    private String name;
    private long totalTime;
    private long remainTime;
    private long comeInTime;

    public Process(String name, long totalTime, long comeInTime) {
        this.name = name;
        this.totalTime = totalTime;
        this.remainTime = totalTime;
        this.comeInTime = comeInTime;
    }

    public void run(long runTime) {
        System.out.println("process " + name + " is running...");
        System.out.println("come in time  : " + comeInTime);
        System.out.println("total time    : " + totalTime);
        System.out.println("remain time   : " + remainTime);
        System.out.println();
        remainTime -= runTime;
        try {
            Thread.sleep(runTime);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public long getTotalTime() {
        return totalTime;
    }

    public long getRemainTime() {
        return remainTime;
    }

    public long getComeInTime() {
        return comeInTime;
    }
}
```

接着创建一个进程等待队列数据结构:

```java
public interface ProcessQueue {

    void add(Process process);

    Process get();

    boolean isEmpty();
}
```

FCFS 算法使用普通的先进先出队列即可实现该进程等待队列:

```java
public class FCFSProcessQueue implements ProcessQueue {

    private Queue<Process> queue = new LinkedList<>();

    @Override
    public void add(Process process) {
        queue.add(process);
    }

    @Override
    public Process get() {
        return queue.poll();
    }

    @Override
    public boolean isEmpty() {
        return queue.isEmpty();
    }
}
```

接下来是调度器的实现,把它设计成可独立运行的 Thread,它对进程等待队列进行轮询,如果有进程的话就从队列中取出一个任务来执行。批处理系统中,调度器调度一个进程都直接让进程执行完毕,也就是给进程运行时间为进程的总时间。

```java
public class Scheduler extends Thread {

    protected ProcessQueue processQueue;

    public Scheduler(ProcessQueue processQueue) {
        this.processQueue = processQueue;
    }
}
```

```java
public class BatchScheduler extends Scheduler {
    public BatchScheduler(ProcessQueue processQueue) {
        super(processQueue);
    }

    @Override
    public void run() {
        while (true) {
            if (!processQueue.isEmpty()) {
                Process process = processQueue.get();
                process.run(process.getTotalTime());
            }
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
```

使用一个模拟器来模拟实际场景下进程的不定时到达,每 1s 到达一个进程,并且进程的运行时间再 0s \~ 3s 之间,使得多个进程会发生竞争关系。

```java
public class ProcessComeEmulator extends Thread {

    private ProcessQueue processQueue;

    public ProcessComeEmulator(ProcessQueue processQueue) {
        this.processQueue = processQueue;
    }

    @Override
    public void run() {
        int processId = 0;
        while (true) {
            System.out.println("process " + processId + " is coming...");
            System.out.println();
            Process process = new Process((processId++) + "", getRandomTime(), System.currentTimeMillis());
            processQueue.add(process);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    private long getRandomTime() {
        return (long) (Math.random() * 3000);
    }
}
```

最后创建一个客户端来运行整个调度程序:

```java
public class FCFSClient {
    public static void main(String[] args) {
        ProcessQueue processQueue = new FCFSProcessQueue();
        ProcessComeEmulator processComeEmulator = new ProcessComeEmulator(processQueue);
        processComeEmulator.start();
        Scheduler scheduler = new FCFSScheduler(processQueue);
        scheduler.start();
    }
}
```

执行结果如下:

```html
process 0 is coming...

process 0 is running...
come in time  : 1528964425691
total time    : 1807
remain time   : 1807

process 1 is coming...

process 1 is running...
come in time  : 1528964426691
total time    : 2311
remain time   : 2311

process 2 is coming...

process 3 is coming...

process 4 is coming...

process 2 is running...
come in time  : 1528964427692
total time    : 2651
remain time   : 2651

process 5 is coming...

process 6 is coming...

process 3 is running...
come in time  : 1528964428692
total time    : 137
remain time   : 137

process 4 is running...
come in time  : 1528964429692
total time    : 2513
remain time   : 2513
```

### 2. SJF

与 FCFS 不同的是,SJF 的进程等待队列需要使用优先级队列来实现:

```java
public class SJFProcessQueue implements ProcessQueue {

    private PriorityQueue<Process> processesQueue = new PriorityQueue<>(
            (o1, o2) -> (int) (o1.getTotalTime() - o2.getTotalTime()));

    @Override
    public void add(Process process) {
        processesQueue.add(process);
    }

    @Override
    public Process get() {
        return processesQueue.poll();
    }

    @Override
    public boolean isEmpty() {
        return processesQueue.isEmpty();
    }
}
```

运行客户端:

```java
public class SJFClient {
    public static void main(String[] args) {
        ProcessQueue processQueue = new SJFProcessQueue();
        ProcessComeEmulator processComeEmulator = new ProcessComeEmulator(processQueue);
        processComeEmulator.start();
        Scheduler scheduler = new BatchScheduler(processQueue);
        scheduler.start();
    }
}
```

```java
process 0 is coming...

process 0 is running...
come in time  : 1528964250005
total time    : 2496
remain time   : 2496

process 1 is coming...

process 2 is coming...

process 1 is running...
come in time  : 1528964251006
total time    : 903
remain time   : 903

process 3 is coming...

process 2 is running...
come in time  : 1528964252006
total time    : 1641
remain time   : 1641

process 4 is coming...

process 5 is coming...

process 4 is running...
come in time  : 1528964254007
total time    : 243
remain time   : 243

process 5 is running...
come in time  : 1528964255007
total time    : 646
remain time   : 646

process 3 is running...
come in time  : 1528964253006
total time    : 2772
remain time   : 2772
```

C
CyC2018 已提交
541
## 进程同步
C
CyC2018 已提交
542

C
CyC2018 已提交
543
### 1. 临界区
C
CyC2018 已提交
544 545 546 547 548 549

对临界资源进行访问的那段代码称为临界区。

为了互斥访问临界资源,每个进程在进入临界区之前,需要先进行检查。

```html
C
CyC2018 已提交
550 551 552
// entry section
// critical section;
// exit section
C
CyC2018 已提交
553 554
```

C
CyC2018 已提交
555
### 2. 同步与互斥
C
CyC2018 已提交
556

C
CyC2018 已提交
557 558
- 同步:多个进程按一定顺序执行;
- 互斥:多个进程在同一时刻只有一个进程能进入临界区。
C
CyC2018 已提交
559

C
CyC2018 已提交
560
### 3. 信号量
C
CyC2018 已提交
561

C
CyC2018 已提交
562
信号量(Semaphore)是一个整型变量,可以对其执行 down 和 up 操作,也就是常见的 P 和 V 操作。
C
CyC2018 已提交
563

C
CyC2018 已提交
564 565
-  **down**  : 如果信号量大于 0 ,执行 -1 操作;如果信号量等于 0,进程睡眠,等待信号量大于 0;
-  **up** :对信号量执行 +1 操作,唤醒睡眠的进程让其完成 down 操作。
C
CyC2018 已提交
566

C
CyC2018 已提交
567
down 和 up 操作需要被设计成原语,不可分割,通常的做法是在执行这些操作的时候屏蔽中断。
C
CyC2018 已提交
568

C
CyC2018 已提交
569
如果信号量的取值只能为 0 或者 1,那么就成为了  **互斥量(Mutex)** ,0 表示临界区已经加锁,1 表示临界区解锁。
C
CyC2018 已提交
570 571

```c
C
CyC2018 已提交
572 573 574 575 576 577
typedef int semaphore;
semaphore mutex = 1;
void P1() {
    down(&mutex);
    // 临界区
    up(&mutex);
C
CyC2018 已提交
578 579
}

C
CyC2018 已提交
580 581 582 583
void P2() {
    down(&mutex);
    // 临界区
    up(&mutex);
C
CyC2018 已提交
584 585 586
}
```

C
CyC2018 已提交
587
<font size=3>  **使用信号量实现生产者-消费者问题**  </font> </br>
C
CyC2018 已提交
588 589 590

问题描述:使用一个缓冲区来保存物品,只有缓冲区没有满,生产者才可以放入物品;只有缓冲区不为空,消费者才可以拿走物品。

C
CyC2018 已提交
591
因为缓冲区属于临界资源,因此需要使用一个互斥量 mutex 来控制对缓冲区的互斥访问。
C
CyC2018 已提交
592

C
CyC2018 已提交
593
为了同步生产者和消费者的行为,需要记录缓冲区中物品的数量。数量可以使用信号量来进行统计,这里需要使用两个信号量:empty 记录空缓冲区的数量,full 记录满缓冲区的数量。其中,empty 信号量是在生产者进程中使用,当 empty 不为 0 时,生产者才可以放入物品;full 信号量是在消费者进程中使用,当 full 信号量不为 0 时,消费者才可以取走物品。
C
CyC2018 已提交
594

C
CyC2018 已提交
595
注意,不能先对缓冲区进行加锁,再测试信号量。也就是说,不能先执行 down(mutex) 再执行 down(empty)。如果这么做了,那么可能会出现这种情况:生产者对缓冲区加锁后,执行 down(empty) 操作,发现 empty = 0,此时生产者睡眠。消费者不能进入临界区,因为生产者对缓冲区加锁了,也就无法执行 up(empty) 操作,empty 永远都为 0,那么生产者和消费者就会一直等待下去,造成死锁。
C
CyC2018 已提交
596 597

```c
C
CyC2018 已提交
598 599 600 601 602 603 604 605 606 607 608 609 610 611 612
#define N 100
typedef int semaphore;
semaphore mutex = 1;
semaphore empty = N;
semaphore full = 0;

void producer() {
    while(TRUE){
        int item = produce_item();
        down(&empty);
        down(&mutex);
        insert_item(item);
        up(&mutex);
        up(&full);
    }
C
CyC2018 已提交
613 614
}

C
CyC2018 已提交
615 616 617 618 619 620 621 622 623
void consumer() {
    while(TRUE){
        down(&full);
        down(&mutex);
        int item = remove_item();
        up(&mutex);
        up(&empty);
        consume_item(item);
    }
C
CyC2018 已提交
624 625 626
}
```

C
CyC2018 已提交
627
### 4. 管程
C
CyC2018 已提交
628 629 630

使用信号量机制实现的生产者消费者问题需要客户端代码做很多控制,而管程把控制的代码独立出来,不仅不容易出错,也使得客户端代码调用更容易。

C
CyC2018 已提交
631
c 语言不支持管程,下面的示例代码使用了类 Pascal 语言来描述管程。示例代码的管程提供了 insert() 和 remove() 方法,客户端代码通过调用这两个方法来解决生产者-消费者问题。
C
CyC2018 已提交
632 633

```pascal
C
CyC2018 已提交
634 635 636 637 638 639 640 641 642 643 644 645 646 647
monitor ProducerConsumer
    integer i;
    condition c;

    procedure insert();
    begin
        // ...
    end;

    procedure remove();
    begin
        // ...
    end;
end monitor;
C
CyC2018 已提交
648 649 650 651
```

管程有一个重要特性:在一个时刻只能有一个进程使用管程。进程在无法继续执行的时候不能一直占用管程,否者其它进程永远不能使用管程。

C
CyC2018 已提交
652
管程引入了  **条件变量**  以及相关的操作:**wait()****signal()** 来实现同步操作。对条件变量执行 wait() 操作会导致调用进程阻塞,把管程让出来给另一个进程持有。signal() 操作用于唤醒被阻塞的进程。
C
CyC2018 已提交
653

C
CyC2018 已提交
654
<font size=3> **使用管程实现生成者-消费者问题** </font><br>
C
CyC2018 已提交
655 656

```pascal
C
CyC2018 已提交
657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681
// 管程
monitor ProducerConsumer
    condition full, empty;
    integer count := 0;
    condition c;

    procedure insert(item: integer);
    begin
        if count = N then wait(full);
        insert_item(item);
        count := count + 1;
        if count = 1 then signal(empty);
    end;

    function remove: integer;
    begin
        if count = 0 then wait(empty);
        remove = remove_item;
        count := count - 1;
        if count = N -1 then signal(full);
    end;
end monitor;

// 生产者客户端
procedure producer
C
CyC2018 已提交
682
begin
C
CyC2018 已提交
683 684 685 686 687
    while true do
    begin
        item = produce_item;
        ProducerConsumer.insert(item);
    end
C
CyC2018 已提交
688 689
end;

C
CyC2018 已提交
690 691
// 消费者客户端
procedure consumer
C
CyC2018 已提交
692
begin
C
CyC2018 已提交
693 694 695 696 697
    while true do
    begin
        item = ProducerConsumer.remove;
        consume_item(item);
    end
C
CyC2018 已提交
698 699 700
end;
```

C
CyC2018 已提交
701
## 经典同步问题
C
CyC2018 已提交
702 703 704

生产者和消费者问题前面已经讨论过了。

C
CyC2018 已提交
705
### 1. 读者-写者问题
C
CyC2018 已提交
706 707 708

允许多个进程同时对数据进行读操作,但是不允许读和写以及写和写操作同时发生。

C
CyC2018 已提交
709
一个整型变量 count 记录在对数据进行读操作的进程数量,一个互斥量 count_mutex 用于对 count 加锁,一个互斥量 data_mutex 用于对读写的数据加锁。
C
CyC2018 已提交
710 711

```c
C
CyC2018 已提交
712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728
typedef int semaphore;
semaphore count_mutex = 1;
semaphore data_mutex = 1;
int count = 0;

void reader() {
    while(TRUE) {
        down(&count_mutex);
        count++;
        if(count == 1) down(&data_mutex); // 第一个读者需要对数据进行加锁,防止写进程访问
        up(&count_mutex);
        read();
        down(&count_mutex);
        count--;
        if(count == 0) up(&data_mutex);
        up(&count_mutex);
    }
C
CyC2018 已提交
729 730
}

C
CyC2018 已提交
731 732 733 734 735 736
void writer() {
    while(TRUE) {
        down(&data_mutex);
        write();
        up(&data_mutex);
    }
C
CyC2018 已提交
737 738 739
}
```

C
CyC2018 已提交
740
### 2. 哲学家进餐问题
C
CyC2018 已提交
741

C
CyC2018 已提交
742
<div align="center"> <img src="../pics//a9077f06-7584-4f2b-8c20-3a8e46928820.jpg"/> </div><br>
C
CyC2018 已提交
743 744 745 746 747 748

五个哲学家围着一张圆桌,每个哲学家面前放着食物。哲学家的生活有两种交替活动:吃饭以及思考。当一个哲学家吃饭时,需要先拿起自己左右两边的两根筷子,并且一次只能拿起一根筷子。

下面是一种错误的解法,考虑到如果所有哲学家同时拿起左手边的筷子,那么就无法拿起右手边的筷子,造成死锁。

```c
C
CyC2018 已提交
749 750 751 752 753 754 755 756 757 758 759
#define N 5

void philosopher(int i) {
    while(TRUE) {
        think();
        take(i);       // 拿起左边的筷子
        take((i+1)%N); // 拿起右边的筷子
        eat();
        put(i);
        put((i+1)%N);
    }
C
CyC2018 已提交
760 761 762 763 764
}
```

为了防止死锁的发生,可以设置两个条件:

C
CyC2018 已提交
765 766
- 必须同时拿起左右两根筷子;
- 只有在两个邻居都没有进餐的情况下才允许进餐。
C
CyC2018 已提交
767 768

```c
C
CyC2018 已提交
769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786
#define N 5
#define LEFT (i + N - 1) % N // 左邻居
#define RIGHT (i + 1) % N    // 右邻居
#define THINKING 0
#define HUNGRY   1
#define EATING   2
typedef int semaphore;
int state[N];                // 跟踪每个哲学家的状态
semaphore mutex = 1;         // 临界区的互斥
semaphore s[N];              // 每个哲学家一个信号量

void philosopher(int i) {
    while(TRUE) {
        think();
        take_two(i);
        eat();
        put_tow(i);
    }
C
CyC2018 已提交
787 788
}

C
CyC2018 已提交
789 790 791 792 793 794
void take_two(int i) {
    down(&mutex);
    state[i] = HUNGRY;
    test(i);
    up(&mutex);
    down(&s[i]);
C
CyC2018 已提交
795 796
}

C
CyC2018 已提交
797 798 799 800 801 802
void put_tow(i) {
    down(&mutex);
    state[i] = THINKING;
    test(LEFT);
    test(RIGHT);
    up(&mutex);
C
CyC2018 已提交
803 804
}

C
CyC2018 已提交
805 806 807 808 809
void test(i) {         // 尝试拿起两把筷子
    if(state[i] == HUNGRY && state[LEFT] != EATING && state[RIGHT] !=EATING) {
        state[i] = EATING;
        up(&s[i]);
    }
C
CyC2018 已提交
810 811 812
}
```

C
CyC2018 已提交
813
## 进程通信
C
CyC2018 已提交
814

C
CyC2018 已提交
815
进程同步与进程通信很容易混淆,它们的区别在于:
C
CyC2018 已提交
816

C
CyC2018 已提交
817 818
- 进程同步:控制多个进程按一定顺序执行;
- 进程通信:进程间传输信息。
C
CyC2018 已提交
819 820 821

进程通信是一种手段,而进程同步是一种目的。也可以说,为了能够达到进程同步的目的,需要让进程进行通信,传输一些进程同步所需要的信息。

C
CyC2018 已提交
822
### 1. 管道
C
CyC2018 已提交
823

C
CyC2018 已提交
824
管道是通过调用 pipe 函数创建的,fd[0] 用于读,fd[1] 用于写。
C
CyC2018 已提交
825

C
CyC2018 已提交
826 827 828 829 830 831
```c
#include <unistd.h>
int pipe(int fd[2]);
```

它具有以下限制:
C
CyC2018 已提交
832

C
CyC2018 已提交
833 834
- 只支持半双工通信(单向传输);
- 只能在父子进程中使用。
C
CyC2018 已提交
835

C
CyC2018 已提交
836
<div align="center"> <img src="../pics//53cd9ade-b0a6-4399-b4de-7f1fbd06cdfb.png"/> </div><br>
C
CyC2018 已提交
837

C
CyC2018 已提交
838
### 2. FIFO
C
CyC2018 已提交
839

C
CyC2018 已提交
840
也称为命名管道,去除了管道只能在父子进程中使用的限制。
C
CyC2018 已提交
841

C
CyC2018 已提交
842 843 844 845 846
```c
#include <sys/stat.h>
int mkfifo(const char *path, mode_t mode);
int mkfifoat(int fd, const char *path, mode_t mode);
```
C
CyC2018 已提交
847

C
CyC2018 已提交
848
FIFO 常用于客户-服务器应用程序中,FIFO 用作汇聚点,在客户进程和服务器进程之间传递数据。
C
CyC2018 已提交
849

C
CyC2018 已提交
850
<div align="center"> <img src="../pics//2ac50b81-d92a-4401-b9ec-f2113ecc3076.png"/> </div><br>
C
CyC2018 已提交
851

C
CyC2018 已提交
852
### 3. 消息队列
C
CyC2018 已提交
853

C
CyC2018 已提交
854
相比于 FIFO,消息队列具有以下优点:
C
CyC2018 已提交
855

C
CyC2018 已提交
856 857 858
- 消息队列可以独立于读写进程存在,从而避免了 FIFO 中同步管道的打开和关闭时可能产生的困难;
- 避免了 FIFO 的同步阻塞问题,不需要进程自己提供同步方法;
- 读进程可以根据消息类型有选择地接收消息,而不像 FIFO 那样只能默认地接收。
C
CyC2018 已提交
859

C
CyC2018 已提交
860
### 4. 信号量
C
CyC2018 已提交
861

C
CyC2018 已提交
862
它是一个计数器,用于为多个进程提供对共享数据对象的访问。
C
CyC2018 已提交
863

C
CyC2018 已提交
864
### 5. 共享存储
C
CyC2018 已提交
865

C
CyC2018 已提交
866
允许多个进程共享一个给定的存储区。因为数据不需要在进程之间复制,所以这是最快的一种 IPC。
C
CyC2018 已提交
867

C
CyC2018 已提交
868
需要使用信号量用来同步对共享存储的访问。
C
CyC2018 已提交
869

C
CyC2018 已提交
870
多个进程可以将同一个文件映射到它们的地址空间从而实现共享内存。另外 XSI 共享内存不是使用文件,而是使用使用内存的匿名段。
C
CyC2018 已提交
871

C
CyC2018 已提交
872
### 6. 套接字
C
CyC2018 已提交
873

C
CyC2018 已提交
874
与其它通信机制不同的是,它可用于不同机器间的进程通信。
C
CyC2018 已提交
875

C
CyC2018 已提交
876
# 三、死锁
C
CyC2018 已提交
877

C
CyC2018 已提交
878
## 死锁的必要条件
C
CyC2018 已提交
879

C
CyC2018 已提交
880
<div align="center"> <img src="../pics//c037c901-7eae-4e31-a1e4-9d41329e5c3e.png"/> </div><br>
C
CyC2018 已提交
881

C
CyC2018 已提交
882 883 884 885
- 互斥:每个资源要么已经分配给了一个进程,要么就是可用的。
- 占有和等待:已经得到了某个资源的进程可以再请求新的资源。
- 不可抢占:已经分配给一个进程的资源不能强制性地被抢占,它只能被占有它的进程显式地释放。
- 环路等待:有两个或者两个以上的进程组成一条环路,该环路中的每个进程都在等待下一个进程所占有的资源。
C
CyC2018 已提交
886

C
CyC2018 已提交
887
## 死锁的处理方法
C
CyC2018 已提交
888

C
CyC2018 已提交
889
### 1. 鸵鸟策略
C
CyC2018 已提交
890 891 892 893 894

把头埋在沙子里,假装根本没发生问题。

因为解决死锁问题的代价很高,因此鸵鸟策略这种不采取任务措施的方案会获得更高的性能。当发生死锁时不会对用户造成多大影响,或发生死锁的概率很低,可以采用鸵鸟策略。

C
CyC2018 已提交
895
大多数操作系统,包括 Unix,Linux 和 Windows,处理死锁问题的办法仅仅是忽略它。
C
CyC2018 已提交
896

C
CyC2018 已提交
897
### 2. 死锁检测与死锁恢复
C
CyC2018 已提交
898 899 900

不试图阻止死锁,而是当检测到死锁发生时,采取措施进行恢复。

C
CyC2018 已提交
901
**(一)每种类型一个资源的死锁检测** 
C
CyC2018 已提交
902

C
CyC2018 已提交
903
<div align="center"> <img src="../pics//b1fa0453-a4b0-4eae-a352-48acca8fff74.png"/> </div><br>
C
CyC2018 已提交
904 905 906

上图为资源分配图,其中方框表示资源,圆圈表示进程。资源指向进程表示该资源已经分配给该进程,进程指向资源表示进程请求获取该资源。

C
CyC2018 已提交
907
图 a 可以抽取出环,如图 b,它满足了环路等待条件,因此会发生死锁。
C
CyC2018 已提交
908 909 910

每种类型一个资源的死锁检测算法是通过检测有向图是否存在环来实现,从一个节点出发进行深度优先搜索,对访问过的节点进行标记,如果访问了已经标记的节点,就表示有向图存在环,也就是检测到死锁的发生。

C
CyC2018 已提交
911
**(二)每种类型多个资源的死锁检测** 
C
CyC2018 已提交
912

C
CyC2018 已提交
913
<div align="center"> <img src="../pics//e1eda3d5-5ec8-4708-8e25-1a04c5e11f48.png"/> </div><br>
C
CyC2018 已提交
914 915 916

上图中,有三个进程四个资源,每个数据代表的含义如下:

C
CyC2018 已提交
917 918 919 920
- E 向量:资源总量
- A 向量:资源剩余量
- C 矩阵:每个进程所拥有的资源数量,每一行都代表一个进程拥有资源的数量
- R 矩阵:每个进程请求的资源数量
C
CyC2018 已提交
921

C
CyC2018 已提交
922
进程 P<sub>1</sub> 和 P<sub>2</sub> 所请求的资源都得不到满足,只有进程 P<sub>3</sub> 可以,让 P<sub>3</sub> 执行,之后释放 P<sub>3</sub> 拥有的资源,此时 A = (2 2 2 0)。P<sub>2</sub> 可以执行,执行后释放 P<sub>2</sub> 拥有的资源,A = (4 2 2 1) 。P<sub>1</sub> 也可以执行。所有进程都可以顺利执行,没有死锁。
C
CyC2018 已提交
923 924 925 926 927

算法总结如下:

每个进程最开始时都不被标记,执行过程有可能被标记。当算法结束时,任何没有被标记的进程都是死锁进程。

C
CyC2018 已提交
928 929 930
1. 寻找一个没有标记的进程 P<sub>i</sub>,它所请求的资源小于等于 A。
2. 如果找到了这样一个进程,那么将 C 矩阵的第 i 行向量加到 A 中,标记该进程,并转回 1。
3. 如果没有这样一个进程,算法终止。
C
CyC2018 已提交
931

C
CyC2018 已提交
932
**(三)死锁恢复** 
C
CyC2018 已提交
933

C
CyC2018 已提交
934 935 936
- 利用抢占恢复
- 利用回滚恢复
- 通过杀死进程恢复
C
CyC2018 已提交
937

C
CyC2018 已提交
938
### 3. 死锁预防
C
CyC2018 已提交
939 940 941

在程序运行之前预防发生死锁。

C
CyC2018 已提交
942
**(一)破坏互斥条件** 
C
CyC2018 已提交
943 944 945

例如假脱机打印机技术允许若干个进程同时输出,唯一真正请求物理打印机的进程是打印机守护进程。

C
CyC2018 已提交
946
**(二)破坏占有和等待条件** 
C
CyC2018 已提交
947 948 949

一种实现方式是规定所有进程在开始执行前请求所需要的全部资源。

C
CyC2018 已提交
950
**(三)破坏不可抢占条件** 
C
CyC2018 已提交
951

C
CyC2018 已提交
952
**(四)破坏环路等待** 
C
CyC2018 已提交
953 954 955

给资源统一编号,进程只能按编号顺序来请求资源。

C
CyC2018 已提交
956
### 4. 死锁避免
C
CyC2018 已提交
957 958 959

在程序运行时避免发生死锁。

C
CyC2018 已提交
960
**(一)安全状态** 
C
CyC2018 已提交
961

C
CyC2018 已提交
962
<div align="center"> <img src="../pics//ed523051-608f-4c3f-b343-383e2d194470.png"/> </div><br>
C
CyC2018 已提交
963

C
CyC2018 已提交
964
图 a 的第二列 Has 表示已拥有的资源数,第三列 Max 表示总共需要的资源数,Free 表示还有可以使用的资源数。从图 a 开始出发,先让 B 拥有所需的所有资源(图 b),运行结束后释放 B,此时 Free 变为 5(图 c);接着以同样的方式运行 C 和 A,使得所有进程都能成功运行,因此可以称图 a 所示的状态时安全的。
C
CyC2018 已提交
965 966 967 968 969

定义:如果没有死锁发生,并且即使所有进程突然请求对资源的最大需求,也仍然存在某种调度次序能够使得每一个进程运行完毕,则称该状态是安全的。

安全状态的检测与死锁的检测类似,因为安全状态必须要求不能发生死锁。下面的银行家算法与死锁检测算法非常类似,可以结合着做参考对比。

C
CyC2018 已提交
970
**(二)单个资源的银行家算法** 
C
CyC2018 已提交
971 972 973

一个小城镇的银行家,他向一群客户分别承诺了一定的贷款额度,算法要做的是判断对请求的满足是否会进入不安全状态,如果是,就拒绝请求;否则予以分配。

C
CyC2018 已提交
974
<div align="center"> <img src="../pics//d160ec2e-cfe2-4640-bda7-62f53e58b8c0.png"/> </div><br>
C
CyC2018 已提交
975

C
CyC2018 已提交
976
上图 c 为不安全状态,因此算法会拒绝之前的请求,从而避免进入图 c 中的状态。
C
CyC2018 已提交
977

C
CyC2018 已提交
978
**(三)多个资源的银行家算法** 
C
CyC2018 已提交
979

C
CyC2018 已提交
980
<div align="center"> <img src="../pics//62e0dd4f-44c3-43ee-bb6e-fedb9e068519.png"/> </div><br>
C
CyC2018 已提交
981

C
CyC2018 已提交
982
上图中有五个进程,四个资源。左边的图表示已经分配的资源,右边的图表示还需要分配的资源。最右边的 E、P 以及 A 分别表示:总资源、已分配资源以及可用资源,注意这三个为向量,而不是具体数值,例如 A=(1020),表示 4 个资源分别还剩下 1/0/2/0。
C
CyC2018 已提交
983 984 985

检查一个状态是否安全的算法如下:

C
CyC2018 已提交
986 987 988
- 查找右边的矩阵是否存在一行小于等于向量 A。如果不存在这样的行,那么系统将会发生死锁,状态是不安全的。
- 假若找到这样一行,将该进程标记为终止,并将其已分配资源加到 A 中。
- 重复以上两步,直到所有进程都标记为终止,则状态时安全的。
C
CyC2018 已提交
989

C
CyC2018 已提交
990
如果一个状态不是安全的,需要拒绝进入这个状态。
C
CyC2018 已提交
991

C
CyC2018 已提交
992
# 四、内存管理
C
CyC2018 已提交
993

C
CyC2018 已提交
994
## 虚拟内存
C
CyC2018 已提交
995

C
CyC2018 已提交
996
虚拟内存的目的是为了让物理内存扩充成更大的逻辑内存,从而让程序获得更多的可用内存。
C
CyC2018 已提交
997

C
CyC2018 已提交
998
为了更好的管理内存,操作系统将内存抽象成地址空间。每个程序拥有自己的地址空间,这个地址空间被分割成多个块,每一块称为一页。这些页被映射到物理内存,但不需要映射到连续的物理内存,也不需要所有页都必须在物理内存中。当程序引用到一部分不在物理内存中的地址空间时,由硬件执行必要的映射,将缺失的部分装入物理内存并重新执行失败的指令。
C
CyC2018 已提交
999

C
CyC2018 已提交
1000
从上面的描述中可以看出,虚拟内存允许程序地址空间中的每一页都映射到物理内存,也就是说一个程序不需要全部调入内存就可以运行,这使得有限的内存运行大程序称为可能。例如有一台计算机可以产生 16 位地址,那么一个程序的地址空间范围是 0\~64K。该计算机只有 32KB 的物理内存,虚拟内存技术允许该计算机运行一个 64K 大小的程序。
C
CyC2018 已提交
1001

C
CyC2018 已提交
1002
<div align="center"> <img src="../pics//7b281b1e-0595-402b-ae35-8c91084c33c1.png"/> </div><br>
C
CyC2018 已提交
1003

C
CyC2018 已提交
1004
## 分页系统地址映射
C
CyC2018 已提交
1005

C
CyC2018 已提交
1006 1007
- 内存管理单元(MMU):管理着地址空间和物理内存的转换。
- 页表(Page table):页(地址空间)和页框(物理内存空间)的映射表。例如下图中,页表的第 0 个表项为 010,表示第 0 个页映射到第 2 个页框。页表项的最后一位用来标记页是否在内存中。
C
CyC2018 已提交
1008

C
CyC2018 已提交
1009
下图的页表存放着 16 个页,这 16 个页需要用 4 个比特位来进行索引定位。因此对于虚拟地址(0010 000000000100),前 4 位是用来存储页面号,而后 12 位存储在页中的偏移量。
C
CyC2018 已提交
1010

C
CyC2018 已提交
1011
(0010 000000000100)根据前 4 位得到页号为 2,读取表项内容为(110 1),它的前 3 为为页框号,最后 1 位表示该页在内存中。最后映射得到物理内存地址为(110 000000000100)。
C
CyC2018 已提交
1012

C
CyC2018 已提交
1013
<div align="center"> <img src="../pics//cf4386a1-58c9-4eca-a17f-e12b1e9770eb.png" width="500"/> </div><br>
C
CyC2018 已提交
1014

C
CyC2018 已提交
1015
## 页面置换算法
C
CyC2018 已提交
1016 1017 1018

在程序运行过程中,如果要访问的页面不在内存中,就发生缺页中断从而将该页调入内存中。此时如果内存已无空闲空间,系统必须从内存中调出一个页面到磁盘对换区中来腾出空间。

C
CyC2018 已提交
1019 1020
页面置换算法和缓存淘汰策略类似。在缓存系统中,缓存的大小有限,当有新的缓存到达时,需要淘汰一部分已经存在的缓存,这样才有空间存放新的缓存数据。

C
CyC2018 已提交
1021 1022
页面置换算法的主要目标是使页面置换频率最低(也可以说缺页率最低)。

C
CyC2018 已提交
1023
### 1. 最佳
C
CyC2018 已提交
1024

C
CyC2018 已提交
1025
> Optimal
C
CyC2018 已提交
1026 1027 1028 1029 1030 1031 1032

所选择的被换出的页面将是最长时间内不再被访问,通常可以保证获得最低的缺页率。

是一种理论上的算法,因为无法知道一个页面多长时间不再被访问。

举例:一个系统为某进程分配了三个物理块,并有如下页面引用序列:

C
CyC2018 已提交
1033
<div align="center"><img src="https://latex.codecogs.com/gif.latex?7,0,1,2,0,3,0,4,2,3,0,3,2,1,2,0,1,7,0,1"/></div> <br>
C
CyC2018 已提交
1034

C
CyC2018 已提交
1035
开始运行时,先将 7, 0, 1 三个页面装入内存。当进程要访问页面 2 时,产生缺页中断,会将页面 7 换出,因为页面 7 再次被访问的时间最长。
C
CyC2018 已提交
1036

C
CyC2018 已提交
1037
### 2. 最近最久未使用
C
CyC2018 已提交
1038

C
CyC2018 已提交
1039
> LRU, Least Recently Used
C
CyC2018 已提交
1040

C
CyC2018 已提交
1041
虽然无法知道将来要使用的页面情况,但是可以知道过去使用页面的情况。LRU 将最近最久未使用的页面换出。
C
CyC2018 已提交
1042

C
CyC2018 已提交
1043 1044 1045 1046 1047
为了实现 LRU,需要在内存中维护一个所有页面的链表。当一个页面被访问时,将这个页面移到链表表头。这样就能保证链表表尾的页面时最近最久未访问的。因为每次访问都需要更新链表,因此这种方式实现的 LRU 代价很高。

<div align="center"><img src="https://latex.codecogs.com/gif.latex?4,7,0,7,1,0,1,2,1,2,6"/></div> <br>

<div align="center"> <img src="../pics//eb859228-c0f2-4bce-910d-d9f76929352b.png"/> </div><br>
C
CyC2018 已提交
1048

C
CyC2018 已提交
1049
### 3. 最近未使用
S
sixzeroo 已提交
1050

C
CyC2018 已提交
1051
> NRU, Not Recently Used
S
sixzeroo 已提交
1052

C
CyC2018 已提交
1053
首先,系统为毎一页面设置了两个状态位。当页面被访问 (读或写) 时设置 R 位; 当页面 (即修改页面) 被写入时设置 M 位。当启动一个进程时,它的所有页面的两个位都由操作系统设置成 0,R 位被定期地 (比如在每次时钟中断时) 清零,以区别最近没有被访问的页面和被访问的页面。
S
sixzeroo 已提交
1054

C
CyC2018 已提交
1055
当发生缺页中断时,操作系统检査所有的页面并根据它们当前的 R 位和 M 位的值,把它们分为 4 类:
S
sixzeroo 已提交
1056

C
CyC2018 已提交
1057 1058 1059 1060
- 第 0 类: 没有被访问,没有被修改
- 第 1 类: 没有被访问,已被修改
- 第 2 类: 已被访问,没有被修改
- 第 3 类: 已被访问,已被修改
S
sixzeroo 已提交
1061

C
CyC2018 已提交
1062
NRU 算法随机地从类编号最小的非空类中挑选一个页面淘汰之。
C
CyC2018 已提交
1063

C
CyC2018 已提交
1064
算法隐含的意思是,在最近一个时钟滴答中 (典型的时间是大约 20ms) 淘汰一个没有被访问的已修改页面要比一个被频繁使用的“十净” 页面好。NRU 主要优点是易于理解和能够有效地被实现,虽然它的性能不是最好的,但是已经够用了。
S
sixzeroo 已提交
1065

C
CyC2018 已提交
1066
### 4. 先进先出
C
CyC2018 已提交
1067

C
CyC2018 已提交
1068
> FIFO, First In First Out
C
CyC2018 已提交
1069

C
CyC2018 已提交
1070
选择换出的页面是最先进入的页面。
C
CyC2018 已提交
1071

C
CyC2018 已提交
1072
该算法会将那些经常被访问的页面也被换出,从而使缺页率升高。
C
CyC2018 已提交
1073

C
CyC2018 已提交
1074
### 5. 第二次机会算法
S
sixzeroo 已提交
1075

C
CyC2018 已提交
1076
FIFO 算法可能会把经常使用的页面置换出去,为了避免这一问题,对该算法做一个简单的修改:
S
sixzeroo 已提交
1077

C
CyC2018 已提交
1078
当页面被访问 (读或写) 时设置该页面的 R 位为 1。需要替换的时候,检查最老页面的 R 位。如果 R 位是 0,那么这个页面既老又没有被使用,可以立刻置换掉;如果是 1,就将 R 位清 0,并把该页面放到链表的尾端,修改它的装入时间使它就像刚装入的一样,然后继续从链表的头部开始搜索
S
sixzeroo 已提交
1079

C
CyC2018 已提交
1080
<div align="center"> <img src="../pics//ecf8ad5d-5403-48b9-b6e7-f2e20ffe8fca.png"/> </div><br>
S
sixzeroo 已提交
1081

C
CyC2018 已提交
1082
第二次机会算法就是寻找一个最近的时钟间隔以来没有被访问过的页面。如果所有的页面都被访问过了,该算法就简化为纯粹的 FIFO 算法。
S
sixzeroo 已提交
1083

C
CyC2018 已提交
1084
### 6. 时钟
C
CyC2018 已提交
1085

C
CyC2018 已提交
1086
> Clock
C
CyC2018 已提交
1087

C
CyC2018 已提交
1088
第二次机会算法需要在链表中移动页面,降低了效率。时钟算法使用环形链表将页面链接起来,再使用一个指针指向最老的页面。
C
CyC2018 已提交
1089

C
CyC2018 已提交
1090
<div align="center"> <img src="../pics//5f5ef0b6-98ea-497c-a007-f6c55288eab1.png"/> </div><br>
C
CyC2018 已提交
1091

C
CyC2018 已提交
1092
## 分段
C
CyC2018 已提交
1093 1094 1095

虚拟内存采用的是分页技术,也就是将地址空间划分成固定大小的页,每一页再与内存进行映射。

C
CyC2018 已提交
1096
下图为一个编译器在编译过程中建立的多个表,有 4 个表是动态增长的,如果使用分页系统的一维地址空间,动态增长的特点会导致覆盖问题的出现。
C
CyC2018 已提交
1097

C
CyC2018 已提交
1098
<div align="center"> <img src="../pics//22de0538-7c6e-4365-bd3b-8ce3c5900216.png"/> </div><br>
C
CyC2018 已提交
1099 1100 1101

分段的做法是把每个表分成段,一个段构成一个独立的地址空间。每个段的长度可以不同,并且可以动态增长。

C
CyC2018 已提交
1102
<div align="center"> <img src="../pics//e0900bb2-220a-43b7-9aa9-1d5cd55ff56e.png"/> </div><br>
C
CyC2018 已提交
1103

C
CyC2018 已提交
1104
## 段页式
C
CyC2018 已提交
1105 1106 1107

程序的地址空间划分成多个拥有独立地址空间的段,每个段上的地址空间划分成大小相同的页。这样既拥有分段系统的共享和保护,又拥有分页系统的虚拟内存功能。

C
CyC2018 已提交
1108
## 分页与分段的比较
C
CyC2018 已提交
1109

C
CyC2018 已提交
1110
- 对程序员的透明性:分页透明,但是分段需要程序员显示划分每个段。
C
CyC2018 已提交
1111

C
CyC2018 已提交
1112
- 地址空间的维度:分页是一维地址空间,分段是二维的。
C
CyC2018 已提交
1113

C
CyC2018 已提交
1114
- 大小是否可以改变:页的大小不可变,段的大小可以动态改变。
C
CyC2018 已提交
1115

C
CyC2018 已提交
1116
- 出现的原因:分页主要用于实现虚拟内存,从而获得更大的地址空间;分段主要是为了使程序和数据可以被划分为逻辑上独立的地址空间并且有助于共享和保护。
C
CyC2018 已提交
1117

C
CyC2018 已提交
1118
# 五、设备管理
C
CyC2018 已提交
1119

C
CyC2018 已提交
1120
## 磁盘调度算法
C
CyC2018 已提交
1121

C
CyC2018 已提交
1122
读写一个磁盘块的时间的影响因素有:
C
CyC2018 已提交
1123

C
CyC2018 已提交
1124 1125 1126 1127 1128
- 旋转时间(主轴旋转磁盘,使得磁头移动到适当的扇区上)
- 寻道时间(制动手臂移动,使得磁头移动到适当的磁道上)
- 实际的数据传输时间

其中,寻道时间最长,因此磁盘调度的主要目标是使磁盘的平均寻道时间最短。
C
CyC2018 已提交
1129

C
CyC2018 已提交
1130
### 1. 先来先服务
C
CyC2018 已提交
1131

C
CyC2018 已提交
1132
> FCFS, First Come First Served
C
CyC2018 已提交
1133

C
CyC2018 已提交
1134 1135 1136
按照磁盘请求的顺序进行调度。

优点是公平和简单。缺点也很明显,因为未对寻道做任何优化,使平均寻道时间可能较长。
C
CyC2018 已提交
1137

C
CyC2018 已提交
1138
### 2. 最短寻道时间优先
C
CyC2018 已提交
1139

C
CyC2018 已提交
1140
> SSTF, Shortest Seek Time First
C
CyC2018 已提交
1141

C
CyC2018 已提交
1142
优先调度与当前磁头所在磁道距离最近的磁道。
C
CyC2018 已提交
1143

C
CyC2018 已提交
1144
虽然平均寻道时间比较低,但是不够公平。如果新到达的磁道请求总是比一个在等待的磁道请求近,那么在等待的磁道请求会一直等待下去,也就是出现饥饿现象。具体来说,两边的磁道请求更容易出现饥饿现象。
C
CyC2018 已提交
1145

C
CyC2018 已提交
1146
<div align="center"> <img src="../pics//4e2485e4-34bd-4967-9f02-0c093b797aaa.png"/> </div><br>
C
CyC2018 已提交
1147

C
CyC2018 已提交
1148
### 3. 电梯算法
C
CyC2018 已提交
1149

C
CyC2018 已提交
1150
> SCAN
C
CyC2018 已提交
1151

C
CyC2018 已提交
1152
电梯总是保持一个方向运行,直到该方向没有请求为止,然后改变运行方向。
C
CyC2018 已提交
1153

C
CyC2018 已提交
1154
电梯算法(扫描算法)和电梯的运行过程类似,总是按一个方向来进行磁盘调度,直到该方向上没有未完成的磁盘请求,然后改变方向。
C
CyC2018 已提交
1155

C
CyC2018 已提交
1156
因为考虑了移动方向,因此所有的磁盘请求都会被满足,解决了 SSTF 的饥饿问题。
C
CyC2018 已提交
1157

C
CyC2018 已提交
1158
<div align="center"> <img src="../pics//271ce08f-c124-475f-b490-be44fedc6d2e.png"/> </div><br>
C
CyC2018 已提交
1159

C
CyC2018 已提交
1160
# 六、链接
C
CyC2018 已提交
1161

C
CyC2018 已提交
1162
## 编译系统
C
CyC2018 已提交
1163

C
CyC2018 已提交
1164
以下是一个 hello.c 程序:
C
CyC2018 已提交
1165 1166

```c
C
CyC2018 已提交
1167
#include <stdio.h>
C
CyC2018 已提交
1168

C
CyC2018 已提交
1169
int main()
C
CyC2018 已提交
1170
{
C
CyC2018 已提交
1171 1172
    printf("hello, world\n");
    return 0;
C
CyC2018 已提交
1173 1174 1175
}
```

C
CyC2018 已提交
1176
在 Unix 系统上,由编译器把源文件转换为目标文件。
C
CyC2018 已提交
1177 1178

```bash
C
CyC2018 已提交
1179
gcc -o hello hello.c
C
CyC2018 已提交
1180 1181 1182 1183
```

这个过程大致如下:

C
CyC2018 已提交
1184
<div align="center"> <img src="../pics//b396d726-b75f-4a32-89a2-03a7b6e19f6f.jpg" width="800"/> </div><br>
C
CyC2018 已提交
1185

C
CyC2018 已提交
1186 1187
- 预处理阶段:处理以 # 开头的预处理命令;
- 编译阶段:翻译成汇编文件;
C
CyC2018 已提交
1188
- 汇编阶段:将汇编文件翻译成可重定向目标文件;
C
CyC2018 已提交
1189
- 链接阶段:将可重定向目标文件和 printf.o 等单独预编译好的目标文件进行合并,得到最终的可执行目标文件。
C
CyC2018 已提交
1190

C
CyC2018 已提交
1191
## 静态链接
C
CyC2018 已提交
1192 1193 1194

静态连接器以一组可重定向目标文件为输入,生成一个完全链接的可执行目标文件作为输出。链接器主要完成以下两个任务:

C
CyC2018 已提交
1195
- 符号解析:每个符号对应于一个函数、一个全局变量或一个静态变量,符号解析的目的是将每个符号引用与一个符号定义关联起来。
C
CyC2018 已提交
1196
- 重定位:链接器通过把每个符号定义与一个内存位置关联起来,然后修改所有对这些符号的引用,使得它们指向这个内存位置。
C
CyC2018 已提交
1197

C
CyC2018 已提交
1198
<div align="center"> <img src="../pics//47d98583-8bb0-45cc-812d-47eefa0a4a40.jpg"/> </div><br>
C
CyC2018 已提交
1199

C
CyC2018 已提交
1200 1201 1202 1203 1204 1205
## 目标文件

- 可执行目标文件:可以直接在内存中执行;
- 可重定向目标文件:可与其它可重定向目标文件在链接阶段合并,创建一个可执行目标文件;
- 共享目标文件:这是一种特殊的可重定向目标文件,可以在加载或者运行时被动态加载进内存并链接;

C
CyC2018 已提交
1206
## 动态链接
C
CyC2018 已提交
1207 1208 1209

静态库有以下两个问题:

C
CyC2018 已提交
1210 1211 1212 1213 1214
- 当静态库更新时那么整个程序都要重新进行链接;
- 对于 printf 这种标准函数库,如果每个程序都要有代码,这会极大浪费资源。

共享库是为了解决静态库的这两个问题而设计的,在 Linux 系统中通常用 .so 后缀来表示,Windows 系统上它们被称为 DLL。它具有以下特点:

C
CyC2018 已提交
1215 1216
- 在给定的文件系统中一个库只有一个文件,所有引用该库的可执行目标文件都共享这个文件,它不会被复制到引用它的可执行文件中;
- 在内存中,一个共享库的 .text 节(已编译程序的机器代码)的一个副本可以被不同的正在运行的进程共享。
C
CyC2018 已提交
1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229

<div align="center"> <img src="../pics//76dc7769-1aac-4888-9bea-064f1caa8e77.jpg"/> </div><br>

# 参考资料

- Tanenbaum A S, Bos H. Modern operating systems[M]. Prentice Hall Press, 2014.
- 汤子瀛, 哲凤屏, 汤小丹. 计算机操作系统[M]. 西安电子科技大学出版社, 2001.
- Bryant, R. E., & O’Hallaron, D. R. (2004). 深入理解计算机系统.
- [Operating System Notes](https://applied-programming.github.io/Operating-Systems-Notes/)
- [进程间的几种通信方式](http://blog.csdn.net/yufaw/article/details/7409596)
- [Operating-System Structures](https://www.cs.uic.edu/\~jbell/CourseNotes/OperatingSystems/2_Structures.html)
- [Processes](http://cse.csusb.edu/tongyu/courses/cs460/notes/process.php)
- [Inter Process Communication Presentation[1]](https://www.slideshare.net/rkolahalam/inter-process-communication-presentation1)