欢迎访问~原文出处——博客园-zhouzhendong

去博客园看该题解


题目传送门 - BZOJ1826


题意概括

  Cache中有m个储存单元,接下来有n个访问地址,每个地址用一个数字表示。访问每一个地址,就要使用一次Cache的一个储存单元,当你选择某一个储存单元时,如果这个储存单元原来不是该地址,那么就发生一次遗失,并把该储存单元的值改为该地址;如果原来这个储存单元就是这个地址,那么不发生遗失且可以直接访问该地址。现在有n个地址访问请求依次输入,每次,你可以选择把地址放在哪一个存储单元,求最少的遗失次数(一开始都没有存储)。


题解

  贪心策略:当有空的存储空间时,先放到空的里面,并记录一次遗失;当每个存储空间都已有记录时,如果该地址已经存在,则直接使用,否则每次替换当前占用的所有地址中下一次出现最晚的,并记录一次遗失。

  对于“下一次出现时间最晚的”,我们用大根堆来维护。


代码

#include <cstring>
#include <algorithm>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <vector>
using namespace std;
const int N=100000+5;
const int inf=1<<28;
vector <int> num[N];
int n,m,a[N],hash[N],hs,size[N],link[N],cnt,ans,heap[N],top,pt[N];
bool f[N];
void read(int &x){
char ch=getchar();
while (!('0'<=ch&&ch<='9'))
ch=getchar();
x=0;
while ('0'<=ch&&ch<='9'){
x=x*10+ch-48;
ch=getchar();
}
}
int find(int x){
int le=1,ri=hs,mid;
while (le<=ri){
mid=(le+ri)/2;
if (hash[mid]==x) return mid;
if (hash[mid]<x) le=mid+1;
if (hash[mid]>x) ri=mid-1;
}
}
void up_sift(int x,int top){
int i=x,j;
while (i>1){
j=i/2;
if (num[heap[i]][link[heap[i]]]<num[heap[j]][link[heap[j]]])
break;
swap(pt[heap[i]],pt[heap[j]]);
swap(heap[i],heap[j]);
i=j;
}
}
void down_sift(int x,int top){
int i=x,j=i*2;
while (j<=top){
if (j<top&&num[heap[j]][link[heap[j]]]<num[heap[j+1]][link[heap[j+1]]])
j++;
if (num[heap[i]][link[heap[i]]]>num[heap[j]][link[heap[j]]])
break;
swap(pt[heap[i]],pt[heap[j]]);
swap(heap[i],heap[j]);
i=j,j=i*2;
}
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++)
scanf("%d",&a[i]),hash[i]=a[i];
sort(hash+1,hash+1+n);
hs=1;
for (int i=2;i<=n;i++)
if (hash[i]!=hash[i-1])
hash[++hs]=hash[i];
for (int i=1;i<=hs;i++)
num[i].clear();
for (int i=1;i<=n;i++)
num[find(a[i])].push_back(i);
for (int i=1;i<=hs;i++)
size[i]=num[i].size(),link[i]=0;
for (int i=1;i<=hs;i++)
num[i].push_back(inf);
memset(f,0,sizeof f),memset(heap,0,sizeof heap),memset(pt,0,sizeof pt);
cnt=ans=top=0;
for (int i=1;i<=n;i++){
int pos=find(a[i]);
link[pos]++;
if (f[pos]){
up_sift(pt[pos],top);
continue;
}
ans++,f[pos]=1;
if (cnt<m){
cnt++,heap[++top]=pos,pt[pos]=top;
up_sift(top,top);
}
else {
f[heap[1]]=0,pt[heap[1]]=0;
heap[1]=pos,pt[pos]=1;
down_sift(1,top);
}
}
printf("%d",ans);
return 0;
}

  

BZOJ1826 [JSOI2010]缓存交换 堆 贪心的更多相关文章

  1. 【BZOJ1826】[JSOI2010]缓存交换(贪心)

    [BZOJ1826][JSOI2010]缓存交换(贪心) 题面 BZOJ 洛谷 题解 当缓存不满显然直接放进去,满了之后考虑拿走哪一个.不难发现拿走下一次出现时间最晚的那个一定不会更差. 那么用一个堆 ...

  2. bzoj1528[POI2005]sam-Toy Cars*&&bzoj1826[JSOI2010]缓存交换

    bzoj1528[POI2005]sam-Toy Cars bzoj1826[JSOI2010]缓存交换 题意: Jasio有n个不同的玩具,它们都被放在了很高的架子上,地板上不会有超过k个玩具.当J ...

  3. [bzoj1826] [JSOI2010]缓存交换

    虽然不知道为什么..但显然,每次扔掉离下次查询最远的内存单元就行了233 用堆来维护贪心...(优先队列大法好 #include<cstdio> #include<iostream& ...

  4. BZOJ_1826_[JSOI2010]缓存交换 _线段树+贪心

    BZOJ_1826_[JSOI2010]缓存交换 _线段树+贪心 Description 在计算机中,CPU只能和高速缓存Cache直接交换数据.当所需的内存单元不在Cache中时,则需要从主存里把数 ...

  5. 1826: [JSOI2010]缓存交换

    1826: [JSOI2010]缓存交换 https://www.lydsy.com/JudgeOnline/problem.php?id=1826 分析: 简单的贪心,然后调啊调...最近怎么了,码 ...

  6. B1826 [JSOI2010]缓存交换 贪心+离散化+堆

    这个题仔细一想可以直接贪心做,因为队列里下一个出现的早的一定最优.正确性显然.然后我只拿了50,我直接模拟另一个队列暴力修改最后一个点的nxt值,自然会T.但是其实不用修改,直接插入就行了前面的不影响 ...

  7. [JSOI2010]缓存交换 贪心 & 堆

    ~~~题面~~~ 题解: 首先我们要使得Miss的次数尽量少,也就是要尽量保证每个点在被访问的时候,这个点已经存在于Cache中. 那么我们可以得到一个结论: 如果Cache已满,那么我们就从Cach ...

  8. JSOI2010 缓存交换

    题目链接:戳我 考虑一个贪心--就是每次我们都选择队列里面之后最晚加入的元素弹出. 维护一个nxt数组就行了. 特判一下之后不会再加入的元素. 代码如下: #include<iostream&g ...

  9. Luogu P4404 [JSOI2010]缓存交换 优先队列

    细节题?...调了半天.... 可以发现,每一次从缓存中删除的主存一定是下次访问最晚的,可以用优先队列来处理...还有要离散化...还有链表末尾要多建一些点...否则会死的很惨... #include ...

随机推荐

  1. 【BZOJ】1443: [JSOI2009]游戏Game

    [算法]博弈论+二分图匹配(最大流) [题解]方格图黑白染色得到二分图, 二分图博弈:当起点不属于某个最大匹配时,后手必胜. 问题转化为那些点不属于某个最大匹配. 先找到一个最大匹配,非匹配点加入答案 ...

  2. Win10 x64 + CUDA 10.0 + cuDNN v7.5 + TensorFlow GPU 1.13 安装指南

    Win10 x64 + CUDA 10.0 + cuDNN v7.5 + TensorFlow GPU 1.13 安装指南 Update : 2019.03.08 0. 环境说明 硬件:Ryzen R ...

  3. JavaScript之能力与硬件检测[小结]

    //检查浏览器种类及其版本号 function checkBrowserVersion(){ var Sys = {}; var ua = navigator.userAgent.toLowerCas ...

  4. WARN bzip2.Bzip2Factory: Failed to load/initialize native-bzip2 library system-native, will use pure-Java version

    [root@hdp2 /root]#hadoop checknative -a 18/12/09 00:31:19 WARN bzip2.Bzip2Factory: Failed to load/in ...

  5. [转]C++11的enum class & enum struct和enum

    1. 旧版enum存在的问题 问题 描述 1 向整形的隐式转换(Implicit conversion to an integer) 2 无法指定底层所使用的数据类型(Inability to spe ...

  6. IMU 预积分推导

    给 StereoDSO 加 IMU,想直接用 OKVIS 的代码,但是有点看不懂.知乎上郑帆写的文章<四元数矩阵与 so(3) 左右雅可比>提到 OKVIS 的预积分是使用四元数,而预积分 ...

  7. Mybatis入门学习笔记

    1.定义别名 在sqlMapConfig.xml中,编写如下代码: <!-- 定义别名 --> <typeAliases> <!-- type: 需要映射的类型 alia ...

  8. HTTP协议中PUT和POST使用上的区别

    有的观点认为,应该用POST来创建一个资源,用PUT来更新一个资源:有的观点认为,应该用PUT来创建一个资源,用POST来更新一个资源:还有的观点认为可以用PUT和POST中任何一个来做创建或者更新一 ...

  9. nvm安装与使用

    1.nvm是什么 nvm全名node.js version management,顾名思义是一个nodejs的版本管理工具.通过它可以安装和切换不同版本的nodejs.下面列出下载.安装及使用方法. ...

  10. MySQL在线更改binlog格式

    今天变更jboss报错如下: SQLWarning ignored: SQL state ', message [Unsafe statement written to the binary log ...