# 时钟页面置换算法(CLOCK)

# 置换策略与算法原理

​ CLOCK页面置换算法的置换策略是淘汰最近未使用的页面,选择在内存中最长时间没有被访问的页面予以淘汰,而替换此时需要调用的页面。

​ CLOCK页面置换算法的算法原理是利用一个循环队列来模拟内存中的页面,并为每个页面设置一个访问位。当某个页面首次装入内存中时,或者该页面被访问到时,将其访问位设置为1。当需要淘汰一个页面时,从当前指针所指的位置开始,检查每个页面的访问位。如果是0,则选择该页面换出,并将新页面放入该位置,并将其访问位设置为1;如果是1,则将其访问位置为0,并继续检查下一个位置。如果一轮扫描后所有页面的访问位都是1,则将所有访问位清零,并重新扫描。

​ CLOCK页面置换算法的优点是实现起来比较简单,只需要维护一个循环队列和一个指针,不需要移动队列中的元素。CLOCK算法也是一种LRU置换算法的近似算法,能够较好地反映程序局部性原理。

扩展:

​ CLOCK页面置换算法的缺点是没有考虑到页面是否被修改过的情况,如果一个页面被修改过,则在换出时需要写回磁盘,这会增加I/O开销。

​ 因此,有一种改进型CLOCK算法,除了访问位外,还增加了一个修改位,用于记录每个页面是否被修改过,并根据不同的情况进行优先级排序。

# 实验步骤

​ 接上文,在这里继续完成置换算法的实现。

// 假设页面的结构体定义如下
struct Page {
  int read; // 读标志
  int write; // 写标志
  int time_arrive; // 到达时间
  int access; // 访问位
};

// 定义一个页面置换算法CLOCK类
class CLOCK {
private:
  int page_num; // 页面数量
  int mem_size; // 内存大小
  Page* pages; // 页面数组
  Page** memory; // 内存数组,用于存放当前正在内存中的页面
  int pointer; // 指针,用于指向当前检查的位置
  int page_faults; // 缺页次数

public:
  // 构造函数,根据给定的页面数量,内存大小和页面数组初始化类的成员变量
  CLOCK(int pn, int ms, Page* ps) {
    page_num = pn;
    mem_size = ms;
    pages = ps;
    memory = new Page*[mem_size]; // 动态分配内存数组的空间
    for (int i = 0; i < mem_size; i++) { // 初始化内存数组,开始时都为空
      memory[i] = nullptr;
    }
    pointer = 0; // 初始化指针为0,指向第一个位置
    page_faults = 0;
  }

  // 析构函数,释放内存数组的空间
  ~CLOCK() {
    delete[] memory;
  }

  // 执行页面置换算法的函数,根据给定的页面访问序列进行操作,并输出结果
  void run(Page* query[], int len) {
    for (int i = 0; i < len; i++) { // 遍历所有页面访问序列
      bool hit = false; // 是否命中标志
      for (int j = 0; j < mem_size; j++) { // 遍历所有内存块
        if (memory[j] == query[i]) { // 如果当前访问的页面已经在内存中
          hit = true; // 设置命中标志为真
          memory[j]->access = 1; // 更新该页面的访问位为1
          break; // 跳出循环
        }
      }
      if (!hit) { // 如果没有命中
        page_faults++; // 缺页次数加一
        bool replaced = false; // 是否替换标志
        while (!replaced) { // 循环直到找到一个可以替换的位置为止
          if (memory[pointer]->access == 0) { // 如果当前指针所指的位置的访问位为0,则可以替换该位置的页面
            memory[pointer] = query[i]; // 将当前访问的页面放入该位置,并将其访问位设置为1
            memory[pointer]->access = 1;
            replaced = true; // 设置替换标志为真,跳出循环
          } else { // 如果当前指针所指的位置的访问位为1,则将其访问位清零,并继续检查下一个位置
            memory[pointer]->access = 0;
            pointer = (pointer + 1) % mem_size; // 指针向前移动一位,如果到达末尾,则回到开头,形成循环队列
          }
        }
      }
    }

    // 输出结果
    cout << "CLOCK page replacement algorithm results:" << endl;
    cout << "Page faults: " << page_faults << endl;
    cout << "Memory contents: " << endl;
    for (int i = 0; i < mem_size; i++) {
      cout << "Memory block " << i << ": ";
      if (memory[i] != nullptr) {
        cout << "Page " << memory[i] - pages << endl;
      } else {
        cout << "Empty" << endl;
      }
    }
  }
};

# 实验样例