BZOJ_2947_[Poi2000]促销_堆
BZOJ_2947_[Poi2000]促销_堆
Description
Bytelandish连锁超市委托你编写一个程序来模拟一项即将施行的促销活动,该活动的规则如下:
●想要参与的顾客,只需把他的个人资料写在帐单上,并把帐单放入投票箱;
●每天活动结束时,数额最大、最小的两张帐单被取出,付款数额最大的顾客将获得一笔奖金,价值为取出的两张帐单的数额之差;
●为了不重复计算,取出的两张帐单不再放回箱子,而剩下的帐单仍保留在箱中,进行第二天的活动。
超市每天的营业额很大,因此可假定:每天活动结束时,箱中至少有两张帐单以供取出。
你的任务是根据每天投入箱中的帐单,计算出这项促销活动期间超市付出的奖金总数额。
任务:
编写一个程序,完成下列工作:
●读入投入箱中的帐单的信息;
●算出促销活动期间的奖金总额;
Input
第一行是一个整数 n(1 <= n <= 5000),表示促销活动历时的天数。
以下的n行,每行包含若干由空格分隔的非负整数。第i+1行的数表示在第i天投入箱子的账单金额。每行的第一行是一个整数k(0 <= k <= 105), 表示当日账单的数目。后面的k个正整数代表这k笔账单的金额,均小于106。
整个活动中涉及到的账单笔数不会超过106 。
Output
唯一一行是一个整数,等于整个促销活动中应该付出的奖金总额。
Sample Input
5
3 1 2 3
2 1 1
4 10 5 5 1
0
1 2
3 1 2 3
2 1 1
4 10 5 5 1
0
1 2
Sample Output
19
可以,不多BB,看代码。
代码:
/*#include <cstdio> #include <cstring> #include <algorithm> using namespace std; inline char nc() { static char buf[100000],*p1,*p2; return p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++; } int rd() { int x=0; char ch=nc(); while(ch<'0'||ch>'9') ch=nc(); while(ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=nc(); return x; } #define N 1000050 #define ls ch[p][0] #define rs ch[p][1] #define get(x) (ch[f[x]][1]==x) long long ans; int ch[N][2],f[N],rt,reimu,val[N],siz[N],n,cnt; int newnode(int v) {val[++reimu]=v; return reimu;} void pushup(int p) {siz[p]=siz[ls]+siz[rs]+1;} void rotate(int x) {int y=f[x],z=f[y],k=get(x);ch[y][k]=ch[x][!k];f[ch[y][k]]=y;ch[x][!k]=y;f[y]=x;f[x]=z;if(z)ch[z][ch[z][1]==y]=x;if(rt==y)rt=x;pushup(y);pushup(x);} void splay(int x,int y) {for(int d;(d=f[x])!=y;rotate(x)) if(f[d]!=y) rotate(get(x)==get(d)?d:x);} void insert(int x) { int l,r,p=rt; while(p) { if(val[p]<x) l=p,p=rs; else r=p,p=ls; } splay(l,0); splay(r,rt); p=newnode(x); ch[r][0]=p; f[p]=r; siz[p]=1; pushup(r); pushup(l); } int pre() {int p=ch[rt][0]; while(rs) p=rs;return p;} int nxt() {int p=ch[rt][1]; while(ls) p=ls;return p;} int find(int x) { int p=rt; while(1) { if(siz[ls]>=x) p=ls; else { x-=siz[ls]+1; if(!x) return p; p=rs; } } } void solve() { int mx,mn,p; p=find(2); mn=val[p]; splay(p,0); int p1=1,p2=nxt(); splay(p1,0); splay(p2,rt); ch[p2][0]=0; pushup(p2); pushup(p1); p=find(cnt); mx=val[p]; splay(p,0); p1=pre(),p2=2; splay(p1,0); splay(p2,rt); ch[p2][0]=0; pushup(p2); pushup(p1); ans=ans+mx-mn; } int main() { scanf("%d",&n); val[1]=-10000000; siz[1]=1; f[1]=2; val[2]=10000000; siz[2]=2; ch[2][0]=1; rt=2; reimu=2; int i,K,x; for(i=1;i<=n;i++) { scanf("%d",&K); while(K--) { scanf("%d",&x); insert(x); cnt++; } solve(); cnt-=2; } printf("%lld\n",ans); }*/ #include <cstdio> #include <cstring> #include <algorithm> #include <ext/pb_ds/priority_queue.hpp> using namespace std; using namespace __gnu_pbds; __gnu_pbds::priority_queue<int>p; __gnu_pbds::priority_queue<int,greater<int> >q; int n,vis1[1000050],vis2[1000050]; long long ans; int main() { scanf("%d",&n); int K,i,x; for(i=1;i<=n;i++) { scanf("%d",&K); while(K--) { scanf("%d",&x); if(vis1[x]) vis1[x]--; else q.push(x); if(vis2[x]) vis2[x]--; else p.push(x); } while(vis1[q.top()]) vis1[q.top()]--,q.pop(); while(vis2[p.top()]) vis2[p.top()]--,p.pop(); ans+=p.top()-q.top(); vis1[p.top()]++; vis2[q.top()]++; q.pop(); p.pop(); } printf("%lld\n",ans); }