BZOJ1826 [JSOI2010]缓存交换 堆 贪心
欢迎访问~原文出处——博客园-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]缓存交换 堆 贪心的更多相关文章
- 【BZOJ1826】[JSOI2010]缓存交换(贪心)
[BZOJ1826][JSOI2010]缓存交换(贪心) 题面 BZOJ 洛谷 题解 当缓存不满显然直接放进去,满了之后考虑拿走哪一个.不难发现拿走下一次出现时间最晚的那个一定不会更差. 那么用一个堆 ...
- bzoj1528[POI2005]sam-Toy Cars*&&bzoj1826[JSOI2010]缓存交换
bzoj1528[POI2005]sam-Toy Cars bzoj1826[JSOI2010]缓存交换 题意: Jasio有n个不同的玩具,它们都被放在了很高的架子上,地板上不会有超过k个玩具.当J ...
- [bzoj1826] [JSOI2010]缓存交换
虽然不知道为什么..但显然,每次扔掉离下次查询最远的内存单元就行了233 用堆来维护贪心...(优先队列大法好 #include<cstdio> #include<iostream& ...
- BZOJ_1826_[JSOI2010]缓存交换 _线段树+贪心
BZOJ_1826_[JSOI2010]缓存交换 _线段树+贪心 Description 在计算机中,CPU只能和高速缓存Cache直接交换数据.当所需的内存单元不在Cache中时,则需要从主存里把数 ...
- 1826: [JSOI2010]缓存交换
1826: [JSOI2010]缓存交换 https://www.lydsy.com/JudgeOnline/problem.php?id=1826 分析: 简单的贪心,然后调啊调...最近怎么了,码 ...
- B1826 [JSOI2010]缓存交换 贪心+离散化+堆
这个题仔细一想可以直接贪心做,因为队列里下一个出现的早的一定最优.正确性显然.然后我只拿了50,我直接模拟另一个队列暴力修改最后一个点的nxt值,自然会T.但是其实不用修改,直接插入就行了前面的不影响 ...
- [JSOI2010]缓存交换 贪心 & 堆
~~~题面~~~ 题解: 首先我们要使得Miss的次数尽量少,也就是要尽量保证每个点在被访问的时候,这个点已经存在于Cache中. 那么我们可以得到一个结论: 如果Cache已满,那么我们就从Cach ...
- JSOI2010 缓存交换
题目链接:戳我 考虑一个贪心--就是每次我们都选择队列里面之后最晚加入的元素弹出. 维护一个nxt数组就行了. 特判一下之后不会再加入的元素. 代码如下: #include<iostream&g ...
- Luogu P4404 [JSOI2010]缓存交换 优先队列
细节题?...调了半天.... 可以发现,每一次从缓存中删除的主存一定是下次访问最晚的,可以用优先队列来处理...还有要离散化...还有链表末尾要多建一些点...否则会死的很惨... #include ...
随机推荐
- 20155334 2016-2017-2 《Java程序设计》第五周学习总结
20155334 2016-2017-2 <Java程序设计>第五周学习总结 教材学习内容总结 第八章:异常处理 Java中所有错误都会被打包为对象,在编程的时候会遇到因各种原因而导致的错 ...
- 第16月第27天 pip install virtualenv ipython sip brew search
1. pip install virtualenv virtualenv testvir cd testvir cd Scripts activate pip https://zhuanlan.zhi ...
- JS 中对变量类型判断的几种方式
文章整理搬运,出处不详,如有侵犯,请联系~ 数据类型判断和数据类型转换代码工具 在 JS 中,有 5 种基本数据类型和 1 种复杂数据类型,基本数据类型有:Undefined, Null, Boo ...
- redis实现消息队列&发布/订阅模式使用
在项目中用到了redis作为缓存,再学习了ActiveMq之后想着用redis实现简单的消息队列,下面做记录. Redis的列表类型键可以用来实现队列,并且支持阻塞式读取,可以很容易的实现一个高性 ...
- 推荐系统之协同过滤的原理及C++实现
1.引言 假如你经营着一家网店,里面卖各种商品(Items),有很多用户在你的店里面买过东西,并对买过的Items进行了评分,我们称之为历史信息,现在为了提高销售量,必须主动向用户推销产品,所以关键是 ...
- Window和document的区别
1.window 窗口对象.就是可视化区域的大小,不包含滚动条内东东. 2.document 对象,包含滚动条以外的区域
- jQuery中【width(),innerWidth(),outerWidth()】
这个问题,已经别扭我多年了,今天终于彻底解决了,拿出来庆贺一下.jquery作为开源项目,无论从思路上,还是从严谨性上,让人崇敬. 随着时间的流逝,jquery的一些功能被逐渐挖掘出来.通过jQuer ...
- 【转】Linux中包管理与定时任务
[转]Linux中包管理与定时任务 第1章 软件查询 1.1 查询软件是否安装 rpm -qa |grep cron 查询是否安装了这个软件. [root@znix ~]# rpm -qa |grep ...
- centos7下配置saltstack
1.下载 使用yum方式即可,可以更换下yum源,这里使用的阿里云的 [root@salt-master ~]# wget -O /etc/yum.repos.d/CentOS-Base.repo h ...
- DDMS调试工具
ADT给我们提供了一个非常方便的调试工具,那就是DDMS.使用这个工具,代码调试工作也变得简单起来.我们只需要单击Eclipse界面右上方的DDMS按钮就可以切换到DDMS界面了,如图2-31所示. ...