51nod 1689 逛街(优先队列)

时间:2023-03-08 19:05:01
51nod 1689 逛街(优先队列)

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1689

题意:

51nod 1689 逛街(优先队列)

题意:

枚举终点,这样就确定路上的花费,接下来只需要计算进店的花费,用三个优先队列维护,q1存储必须要进的ci为1的k个店的最小进店花费,q2存储除了q1中的店之外还能进的店,q3存储暂时不能进的店。

 #include<iostream>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<vector>
#include<stack>
#include<queue>
#include<cmath>
#include<map>
#include<set>
using namespace std;
typedef long long ll;
typedef pair<int,int> pll;
const int INF = 0x3f3f3f3f;
const int maxn = +; int n, k, t;
int a[maxn],b[maxn],c[maxn]; priority_queue<int> q1,q2,q3; int main()
{
//freopen("in.txt","r",stdin);
while(~scanf("%d%d%d",&n,&t,&k))
{
for(int i=;i<=n;i++) scanf("%d",&a[i]);
for(int i=;i<=n;i++) scanf("%d",&b[i]);
for(int i=;i<=n;i++) scanf("%d",&c[i]); ll sum1=,sum2=;
int ans=-;
for(int i=;i<=n;i++)
{
if(c[i])
{
q1.push(b[i]);
sum1+=b[i];
}
else
{
q2.push(b[i]);
sum2+=b[i];
}
if(q1.size()>k)
{
int tmp = q1.top();
sum1-=tmp;
sum2+=tmp;
q2.push(tmp);
q1.pop();
}
if(q1.size()<k) continue;
if(sum1+a[i]>t) continue;
ll left=t-sum1-a[i];
while(!q3.empty() && !q2.empty() && -q3.top()<q2.top()) //q2的最大花费和q3的最小花费交换
{
int tmp2=q2.top();
int tmp3=q3.top();
q2.pop();
q3.pop();
sum2-=tmp2;
sum2-=tmp3;
q2.push(-tmp3);
q3.push(-tmp2);
}
while(!q2.empty() && sum2>left) //如果q2里的进店花费超过了限制,则需要去掉一些
{
int tmp=q2.top();
q2.pop();
sum2-=tmp;
q3.push(-tmp);
}
while (!q3.empty() && left >= sum2 - q3.top()) //q2加上q3里较小的还是满足的
{
q2.push(-q3.top());
sum2 -= q3.top();
q3.pop();
}
if(sum1+sum2+a[i]<=t && k+(int)q2.size()>ans)
ans=k+(int)q2.size();
}
printf("%d\n",ans);
}
return ;
}