[bzoj1122][POI2008]账本BBB

1122: [POI2008]账本BBB

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 402  Solved: 202
[Submit][Status][Discuss]

Description

一个长度为n的记账单,+表示存¥1,-表示取¥1。现在发现记账单有问题。一开始本来已经存了¥p,并且知道最后账户上还有¥q。你要把记账单修改正确,使得 1:账户永远不会出现负数; 2:最后账户上还有¥q。你有2种操作: 1:对某一位取反,耗时x; 2:把最后一位移到第一位,耗时y。

Input

The first line contains 5 integers n, p, q, x and y (1 n 1000000, 0 p;q 1000000, 1 x;y 1000), separated by single spaces and denoting respectively: the number of transactions done by Byteasar, initial and final account balance and the number of seconds needed to perform a single turn (change of sign) and move of transaction to the beginning. The second line contains a sequence of n signs (each a plus or a minus), with no spaces in-between. 1 ≤ n ≤ 1000000, 0 ≤ p ,q ≤ 1000000, 1 ≤x,y ≤ 1000)

Output

修改消耗的时间

Sample Input

9 2 3 2 1
---++++++

Sample Output

3

HINT

Source

老师的惊天模拟赛#2

TAT还是不会

相当神的题目

记录一下序列前缀和

若p+序列和≠q,可以发现取反操作数量是确定的

且尽量在前面做加法,后面做减法

至于旋转操作,暴力想法是把最后一位提前,就相当于连成一个环,在环上求值

所以在环上枚举起点,就相当于移动操作

要是移动不能满足非负,还可以把前面的-操作改为+,对应的后面的+改为-

但暴力走一遍环单纯是为了解决非负的问题

而对于一个起点的序列,若已知其最小值并把它修改至大于0,则显然前后值都不会小于零(想象前缀和)

所以在环上用单调队列求一遍最小值,再枚举起点做无旋转的修改,求最小花费

而题目保证有解题目保证有解题目保证有解

所以除了必要的修改,还需要前后取反时,使最终序列和满足要求的操作一定已经用完了(不论是+变-还是-变+)

那么直接在最小值上加上取反得到的值(一定要大于0),如果还小的话再前后取反

(要是必要的操作是减变加,因为题目有解,所以修改操作一定不在最小值位置之前)

还有好多细节,看代码

 1
 2 #include<cstdio>
 3 #include<cstdlib>
 4 #include<cstring>
 5 #include<algorithm>
 6 #define LL long long
 7 int que[2000020],seq[2000020],mn[1000010];
 8 char opt[1000002];
 9 inline LL abs(LL a){
10     return a>0?a:-a;
11 }
12 inline LL max(LL a,LL b){
13     return a>b?a:b;
14 }
15 inline LL min(LL a,LL b){
16     return a<b?a:b;
17 }
18 int main(){
19     LL n,p,q,x,y;
20     int i,h,t;
21     scanf("%lld %lld %lld %lld %lld",&n,&p,&q,&x,&y);
22     scanf("%s",opt+1);
23     for(i=n<<1;i>n;--i)seq[i]=seq[i+1]+(opt[i-n]==‘+‘?1:-1);
24     for(i=n;i;--i)seq[i]=seq[i+1]+(opt[i]==‘+‘?1:-1);
25     que[0]=0;
26     h=1,t=0;
27     for(i=n<<1;i;--i){
28         while(h<=t&&seq[i]>seq[que[t]])--t;//seq[i]-seq[que[h]]>seq[que[t]]-seq[que[h]]
29         que[++t]=i;
30         while(h<=t&&que[h]-i>=n)++h;
31         if(i<=n)mn[i]=seq[i]-seq[que[h]]; //你不能用head更新min,只能倒序用i
32     }
33     LL all=seq[n+1],tmp=(q-p-all)/2,ans=1e16,cst;
34     for(i=0;i<n;i++){//枚举起点与枚举旋转次数不同
35         cst=x*abs(tmp)+y*(LL)i;
36         if(i==0){
37             mn[1]+=p+max(tmp,0)*2;
38             if(mn[1]<0)cst+=2*x*((1-mn[1])/2);
39         }else{
40             mn[n-i+1]+=p+max(tmp,0)*2;
41             if(mn[n-i+1]<0)cst+=2*x*((1-mn[n-i+1])/2);
42         }
43         ans=min(ans,cst);
44     }
45     printf("%lld",ans);
46     return 0;
47 }

时间: 2024-08-19 14:53:13

[bzoj1122][POI2008]账本BBB的相关文章

BZOJ 1122 POI2008 账本BBB 单调队列

题目大意:给定一个由+1和?1构成的长度为n的序列,提供两种操作: 1.将某一位取反,花销为x 2.将最后一位移动到前一位.花销为y 要求终于p+sumn=q.且p+sumi≥0(1≤i≤n),求最小花销 枚举终于的序列以哪个点開始.那么从这个点往后的最小前缀和能够用单调队列预处理出来 然后贪心地把左边的?1改成+1.右边的+1改成?1直到满足要求就可以 #include <cstdio> #include <cstring> #include <iostream> #

省选之前的未完成的计划(截至到省选)

PLAN OF THE COMING HEOI good problems:-bzoj4823:[Cqoi2017]老C的方块 [*]-bzoj3171:[Tjoi2013]循环格 [*]-bzoj4200:[Noi2015]小园丁与老司机 [*]-bzoj1061:[Noi2008]志愿者招募 [*]-bzoj3600:没有人的算术 [*]-bzoj2806:[Ctsc2012]Cheat [*]-bzoj2219:数论之神 [*]-bzoj2595:[Wc2008]游览计划 [*]-bzoj

波兰题目补全计划

Introduce 本人比较喜欢做波兰的题目,感觉这些题目十分清真,思维也比较好.欢迎同样喜欢波兰题目的OIer来交流.以下是我有记录地刷过的题目. 比较好的题吧:BZOJ #3746.[POI2015]Czarnoksi??nicy okr?g?ego sto?u source:XXII OI - Etap I - Zadanie Czarnoksi??nicy okr?g?ego sto?u notes: 动态规划我的题解http://www.cnblogs.com/TSHugh/p/882

POI2008 题解

POI2008 完结(′?_?`)撒花! 海报PLA 单调栈裸题! 激光发射器SZK 光路可逆? 然后证一下发射器与接收器两两对应? 砖块Klo 区间中值! 可用树状数组水过... 将高度 \( h \) 的值域作为树状数组维护的序列,维护一下前缀数量与前缀和即可. BLO 求割点裸题! Sta so easy 的树上DP! CLO 蜜汁构造... Tro 草稿题... 把差积分开算,求和 枪战Maf 有向仙人掌? naive! 贪心构造即可(<--口胡的) 正解还要分好多种情况? 账本BBB

区块链组织-超级账本(Hyperledger)的简介

区块链上有国外比较著名的组织比如R3联盟.超级账本(Hyperledger),国内的中国区块链研究联盟(China Blockchain Research Alliance).中国分布式总账基础协议联盟(China Ledger联盟).金融区块链合作联盟等. 超级账本(Hyperledger)是Linux基金会于2015年发起的推进区块链数字技术和交易验证的开源项目,目标是让成员共同合作,共建开放平台,满足来自多个不同行业各种用户案例,并简化业务流程.创始人有IBM.Intel.思科等大公司.2

BZOJ1124: [POI2008]枪战Maf

1124: [POI2008]枪战Maf Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 501  Solved: 200[Submit][Status][Discuss] Description 有n个人,每个人手里有一把手枪.一开始所有人都选定一个人瞄准(有可能瞄准自己).然后他们按某个顺序开枪,且任意时刻只有一个人开枪.因此,对于不同的开枪顺序,最后死的人也不同. Input 输入n人数<1000000 每个人的aim Output 你要求最

1131: [POI2008]Sta

1131: [POI2008]Sta Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 783  Solved: 235[Submit][Status] Description 给出一个N个点的树,找出一个点来,以这个点为根的树时,所有点的深度之和最大 Input 给出一个数字N,代表有N个点.N<=1000000 下面N-1条边. Output 输出你所找到的点,如果具有多个解,请输出编号最小的那个. Sample Input 8 1 4 5 6

BZOJ 1113: [Poi2008]海报PLA

题目 1113: [Poi2008]海报PLA Time Limit: 10 Sec  Memory Limit: 162 MB Description N个矩形,排成一排. 现在希望用尽量少的矩形海报Cover住它们. Input 第一行给出数字N,代表有N个矩形.N在[1,250000] 下面N行,每行给出矩形的长与宽.其值在[1,1000000000]2 1/2 Postering Output 最少数量的海报数. Sample Input 5 1 2 1 3 2 2 2 5 1 4 Sa

AC日记——[POI2008]BLO-Blockade 洛谷 [POI2008]BLO-Blockade

[POI2008]BLO-Blockade 思路: tarjan: 代码: #include <bits/stdc++.h> using namespace std; #define maxn 100005 #define maxm 500005 #define ll long long ll Count[maxn],E[maxm<<1],V[maxm<<1],head[maxn],cnt; ll size[maxm],dfn[maxn],n,m,tot,low[max