C++实现虚拟内存页面置换算法(FIFO, OPT, LRU)

虚拟内存页面置换算法(FIFO, OPT, LRU)

0x01 FIFO

  • 置换策略:置换掉先来的页面(FIFO队列首元素)
  • 优点: 简单易理解且易实现
  • 缺点: 性能不理想,会发生Belady异常(页框(帧)增加缺页率也增加)

0x02 OPT

最优页面置换算法(optimal page-replacement algorithm)

  • 置换策略:根据以后使用情况置换:
    置换掉以后不再用;如果没有不再使用的,置换掉相对最迟使用的。
  • 优点:它是最优的页面置换算法,不会发生Belady异常
  • 缺点:需要知道以后页面使用情况,实现起来困难

0x03 LRU

最近最少使用算法(Least-Recent-Used algorithm)

  • 置换策略:根据最近使用情况(上次使用时间)置换:置换掉相对当前时刻最长时间没有使用的页面
    (相对而言,最近刚用的不置换)
  • 优点:近似最优,没有Belady异常
  • 缺点:可能需要重要的硬件辅助,实现上比较困难

代码

以下根据个人思路所写,仅供参考

#include <iostream>
#include <cstring>
#include <algorithm>
#include <queue>
using namespace std;

const int MAX_SIZE = 1e3+5;
const  int INF = 0x3f3f3f3f;
int pg_last[MAX_SIZE]; //last time of using page
bool vis[MAX_SIZE]; //marked in queue

class Node
{
public:
    int node[MAX_SIZE];
    int iSize;
    int cur;
    bool isFull();
    bool exist(int x);
    Node()
    {
        cur = 0;
    }
    Node(int sz)
    {
        iSize = sz;
        cur = 0;
    }
};

// struct Pair 辅助OPT
struct Pair
{
    int pos;
    int id;
    bool operator<(const Pair x)const
    {
        return id > x.id;
    }
} op[MAX_SIZE];


bool Node::isFull()
{
    return cur == iSize;
}

void createPages(int n,Node &pg)
{
    cout<<"请输入长度为"<<n<<"的页面访问序列:";
    pg.iSize = n;
    for(int i = 0; i < n; ++i)
    {
        cin>>pg.node[i];
    }
}

void printResult(int n,int fail_cnt,int replace_cnt)
{
    double fail_rate = fail_cnt / double(n) * 100;
    cout<<" fail times: "<<fail_cnt<<" fail_rate: "<<fail_rate<<"%"<<endl;
    cout<<" replace times: "<<replace_cnt<<endl;
    cout<<"--------------------------------------"<<endl;
}

bool Node::exist(int x)
{
    for(int i = 0; i < cur; ++i)
    {
        if(x == node[i])
        {
            return true;
        }
    }
    return false;
}

void FIFO(int m,Node pg)
{
    Node st(m); // storage
    int fail_cnt = 0;
    int replace_cnt = 0;
    queue<int>q;
    memset(vis,false,sizeof(vis));
    for(int i = 0; i < pg.iSize; ++i)
    {
        if(vis[pg.node[i]]) continue;
        if(int(q.size()) < m)
        {
            q.push(pg.node[i]);
            vis[pg.node[i]] = true;
            fail_cnt++;
        }
        else
        {
            int aim = q.front();
            q.pop();
            vis[aim] = false;
            q.push(pg.node[i]);
            vis[pg.node[i]] = true;
            fail_cnt++;
            replace_cnt++;
        }
    }
    cout<<"-------------FIFO Result--------------"<<endl;
    printResult(pg.iSize,fail_cnt,replace_cnt);
}

int findRepOPT(int s,Node st,Node pg)
{
    int k = 0;
    bool flag; // 标记以后是否会使用
    for(int i = 0; i < st.iSize; ++i)
    {
        flag = false;
        for(int j = s; j < pg.iSize; ++j)
        {
            if(st.node[i] == pg.node[j])
            {
                flag = true;
                op[k].id = j;
                op[k++].pos = i;
                break;
            }
        }
        if(!flag)
        {
            return i;
        }
    }
    sort(op,op+k);
    return op[0].pos;
}

void OPT(int m,Node pg)
{
    Node st(m); // storage
    int fail_cnt = 0;
    int replace_cnt = 0;
    for(int i = 0; i < pg.iSize; ++i)
    {
        if(st.exist(pg.node[i])) continue;
        if(!st.isFull())
        {
            st.node[st.cur++] = pg.node[i];
            fail_cnt++;
        }
        else
        {
            int pos = findRepOPT(i,st,pg);
            st.node[pos] = pg.node[i];
            fail_cnt++;
            replace_cnt++;
        }
    }
    cout<<"-------------OPT Result--------------"<<endl;
    printResult(pg.iSize,fail_cnt,replace_cnt);
}

int findRepLRU(Node st)
{
    int minx = INF,pos = 0;
    for(int i = 0; i < st.iSize; ++i)
    {
        if(pg_last[st.node[i]] < minx)
        {
            minx = pg_last[st.node[i]];
            pos = i;
        }
    }
    return pos;
}

void LRU(int m,Node pg)
{
    Node st(m); // storage
    int fail_cnt = 0;
    int replace_cnt = 0;
    for(int i = 0; i < pg.iSize; ++i)
    {
        pg_last[pg.node[i]] = i;
        if(st.exist(pg.node[i])) continue;
        if(!st.isFull())
        {
            st.node[st.cur++] = pg.node[i];
            fail_cnt++;
        }
        else
        {
            int pos = findRepLRU(st);
            st.node[pos] = pg.node[i];
            fail_cnt++;
            replace_cnt++;
        }
    }
    cout<<"-------------LRU Result--------------"<<endl;
    printResult(pg.iSize,fail_cnt,replace_cnt);

}


int main()
{
    /*
    3 12
    2 3 2 1 5 2 4 5 3 2 5 2
    */
    /*
    3 20
    7 0 1 2 0 3 0 4 2 3 0 3 2 1 2 0 1 7 0 1
    */
    int m,n;
    cout<<"请输入分配给进程的页框数 和 进程页面数:";
    cin>>m>>n;
    Node pg;
    createPages(n,pg);
    cout<<"-------------选择页面置换算法---------------"<<endl;
    cout<<"-------------1.FIFO---------------"<<endl;
    cout<<"-------------2.OPT---------------"<<endl;
    cout<<"-------------3.LRU---------------"<<endl;
    cout<<"-------------0.退出---------------"<<endl;
    int op;
    while(cin>>op)
    {
        switch(op)
        {
        case 1:
            FIFO(m,pg);
            break;
        case 2:
            OPT(m,pg);
            break;
        case 3:
            LRU(m,pg);
            break;
        case 0:
            exit(0);
        }
    }

    return 0;
}

运行实例

 3 12
 2 3 2 1 5 2 4 5 3 2 5 2

在这里插入图片描述

3 20
 7 0 1 2 0 3 0 4 2 3 0 3 2 1 2 0 1 7 0 1

在这里插入图片描述

时钟算法 页面置换算法 #pragma once #include <iostream> #include "Pclass.h" using namespace std; class Allocation { private: int pnum;//页面数 int bnum;//分配块数 //int ID; int num;//访问页面次数 Pclass * block;//块数组 int *page;//页访问顺序 //int p; public: Allocation(int pn,int bl,int n) { pnum=pn; bnum=bl; num=n; //ID=id; block=new Pclass [bnum]; page=new int [num]; cout<<"页面访问顺序:"; for(int i=0;i<num;i++) { page[i]=rand()%pnum+1; cout<<page[i]<<" "; } cout<<endl; //p=0; } void FIFO() { cout<<"先进先出算法:"<<endl; int p=0; int nm=0; for(int i=0;i<num;i++) { if(Haven(page[i])!=(-1)) { nm++; } else { block[p].num=page[i]; p=(++p)%bnum; } for(int j=0;j<bnum;j++) { if(block[j].num!=0) { cout<<block[j].num<<" "; } } cout<<endl; } cout<<"命中率为:"<<(double)nm/(double)num<<endl; } int Haven(int n) { for(int i=0;i<bnum;i++) { if(block[i].num==n) { return i; } } return -1; } void LRU() { int p=0; int nm=0; for(int i=0;i<bnum;i++) { block[i].num=0; block[i].visited=false; } for(int i=0;i<num;i++) { int temp=Haven(page[i]); if(temp!=(-1)) { block[temp].visited=true; nm++; } else { //int j=0; while(1) { if(block[p].visited==false) { block[p].num=page[i]; p=(++p)%bnum; break; } else { block[p].visited=false; p=(++p)%bnum; } } } for(int j=0;j<bnum;j++) { if(block[j].num!=0) { cout<<block[j].num<<" "; } } cout<<endl; } cout<<"命中率为:"<<(double)nm/(double)num<<endl; } void SLRU() { int p=0; int nm=0; for(int i=0;i<bnum;i++) { block[i].num=0; block[i].visited=false; } for(int i=0;i<num;i++) { int temp=Haven(page[i]); if(temp!=(-1)) { block[temp].visited=true; nm++; } else { for(int j=0;j<bnum;j++) { if(block[p].visited==false&&block[p].changed==false) { block[p].num=page[i]; if(rand()%2) { block[p].changed=true; } p=(++p)%bnum; goto over; } else { block[p].visited=false; p=(++p)%bnum; } } while(1) { if(block[p].changed==false) { block[p].num=page[i]; p=(++p)%bnum; break; } else { block[p].changed=false; p=(++p)%bnum; } } } over: for(int j=0;j<bnum;j++) { if(block[j].num!=0) { cout<<block[j].num<<" "; } } cout<<endl; } cout<<"命中率为:"<<(double)nm/(double)num<<endl; } ~Allocation(void) { } };
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Leo Bliss

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值