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 ...
随机推荐
- Python3.X爬虫
1.Python很有名,但是一直没在实际项目中用过,今天花30分钟学习下.去Python官网https://www.python.org/downloads/ 2.2.X与3.X版本相差比较大,新手用 ...
- GBK-----UTF-8编码格式问题浅析
首先,想必大家在不同环境下切换写代码的时候,都会遇见乱码的时候(读取二进制的时候采用的编码和最初将字符转换成二进制时的编码不一致.),大多数人都知道,只需要把项目工程的编码格式调整一下为最初的编码就可 ...
- CentOS6.8配置SonarQube Scanner配合SonarQube使用
下载最新的SonarQube Scanner压缩包 https://docs.sonarqube.org/display/SCAN/Analyzing+with+SonarQube+Scanner ...
- python set()、len()、type()、保留小数、EOFError
1.集合set() set()存的数据为不重复的,可以用此来过滤重复的元素.可以用&求两个set的交集 c = a&b,用|求两个set的并集 c = a | b 用for i in ...
- C 语言 register 关键字
register:这个关键字请求编译器尽可能的将变量存在CPU内部寄存器中,而不是通过内存寻址访问,以提高效率.注意是尽可能,不是绝对.你想想,一个CPU 的寄存器也就那么几个或几十个,你要是定义了很 ...
- MFC_CFileDialog_选择单一文件
场景 选择单一文件 技术点 CFileDialog CFileDialog::CFileDialog( BOOL bOpenFileDialog, LPCTSTR lpszDefExt = NULL, ...
- mysql中文乱码或提示error
插入一条中文记录: 语句: insert into employee(id,name,job,salary) values(4,'小明','清洁员',1500); 提示: ERROR 1366 (HY ...
- 企业内部在centos7.2系统中必杀技NTP时间服务器及内网服务器时间同步(windows和linux客户端同步)
网络时间协议NTP(Network Time Protocol)是用于互联网中时间同步的标准互联网协议.NTP的用途是把计算机的时间同步到某些时间标准.目前采用的时间标准是世界协调时UTC(Unive ...
- sklearn调参(验证曲线,可视化不同参数下交叉验证得分)
一 . 原始方法: 思路: 1. 参数从 0+∞ 的一个 区间 取点, 方法如: np.logspace(-10, 0, 10) , np.logspace(-6, -1, 5) 2. 循环调用cr ...
- PYTHON-模块 sys os random shutil
import sys # 环境变量# print(sys.path)# # 查看已经加载的模块# print(sys.modules)# # 获取终端调用时的参数# print(sys.argv)# ...