Description

农夫Byteasar买了一片n亩的土地,他要在这上面种草。
他在每一亩土地上都种植了一种独一无二的草,其中,第i亩土地的草每天会长高a[i]厘米。
Byteasar一共会进行m次收割,其中第i次收割在第d[i]天,并把所有高度大于等于b[i]的部分全部割去。Byteasar想知道,每次收割得到的草的高度总和是多少,你能帮帮他吗?
 

Input

第一行包含两个正整数n,m(1<=n,m<=500000),分别表示亩数和收割次数。
第二行包含n个正整数,其中第i个数为a[i](1<=a[i]<=1000000),依次表示每亩种植的草的生长能力。
接下来m行,每行包含两个正整数d[i],b[i](1<=d[i]<=10^12,0<=b[i]<=10^12),依次描述每次收割。
数据保证d[1]<d[2]<...<d[m],并且任何时刻没有任何一亩草的高度超过10^12。
 

Output

输出m行,每行一个整数,依次回答每次收割能得到的草的高度总和。
 

Sample Input

4 4
1 2 4 3
1 1
2 2
3 0
4 4

Sample Output

6
6
18
0

HINT

第1天,草的高度分别为1,2,4,3,收割后变为1,1,1,1。
第2天,草的高度分别为2,3,5,4,收割后变为2,2,2,2。
第3天,草的高度分别为3,4,6,5,收割后变为0,0,0,0。
第4天,草的高度分别为1,2,4,3,收割后变为1,2,4,3。
 
不难发现无论何时草高度从小到大的顺序是不变的,我们可以按生长速度先排序一下。
然后问题就比较简单了,一次收割其实类似一个区间覆盖,我们只需找到第一个被收割的草即可。
用线段树维护区间最后一棵草上次被收割的时间、上次剩多长、所有草高度和减去时间*生长高度之和,很容易就O(logn)了。
  1. #include<cstdio>
  2. #include<cctype>
  3. #include<queue>
  4. #include<cstring>
  5. #include<algorithm>
  6. #define rep(i,s,t) for(int i=s;i<=t;i++)
  7. #define dwn(i,s,t) for(int i=s;i>=t;i--)
  8. #define ren for(int i=first[x];i;i=next[i])
  9. using namespace std;
  10. const int BufferSize=1<<16;
  11. char buffer[BufferSize],*head,*tail;
  12. inline char Getchar() {
  13. if(head==tail) {
  14. int l=fread(buffer,1,BufferSize,stdin);
  15. tail=(head=buffer)+l;
  16. }
  17. return *head++;
  18. }
  19. typedef long long ll;
  20. inline ll read() {
  21. ll x=0,f=1;char c=Getchar();
  22. for(;!isdigit(c);c=Getchar()) if(c=='-') f=-1;
  23. for(;isdigit(c);c=Getchar()) x=x*10+c-'0';
  24. return x*f;
  25. }
  26. const int maxn=500010;
  27. int n,m,A[maxn];
  28. ll setv[maxn<<2],sett[maxn<<2],suma[maxn<<2],sumv[maxn<<2],maxv[maxn<<2],maxt[maxn<<2];
  29. void maintain(int o,int l,int r) {
  30. int lc=o<<1,rc=lc|1;
  31. if(setv[o]>=0) {
  32. sumv[o]=setv[o]*(r-l+1)-sett[o]*suma[o];
  33. maxv[o]=setv[o];maxt[o]=sett[o];
  34. }
  35. else if(l<r) {
  36. sumv[o]=sumv[lc]+sumv[rc];
  37. maxv[o]=maxv[rc];maxt[o]=maxt[rc];
  38. }
  39. }
  40. void pushdown(int o,int l,int r) {
  41. int lc=o<<1,rc=lc|1,mid=l+r>>1;
  42. if(setv[o]>=0) {
  43. setv[lc]=setv[rc]=setv[o];
  44. sett[lc]=sett[rc]=sett[o];
  45. maxt[lc]=maxt[rc]=sett[o];
  46. sumv[lc]=setv[o]*(mid-l+1)-sett[o]*suma[lc];
  47. sumv[rc]=setv[o]*(r-mid)-sett[o]*suma[rc];
  48. maxv[lc]=maxv[rc]=setv[o];
  49. setv[o]=sett[o]=-1;
  50. }
  51. }
  52. void build(int o,int l,int r) {
  53. if(l==r) suma[o]=A[l];
  54. else {
  55. setv[o]=-1;
  56. int mid=l+r>>1,lc=o<<1,rc=lc|1;
  57. build(lc,l,mid);build(rc,mid+1,r);
  58. suma[o]=suma[lc]+suma[rc];
  59. }
  60. }
  61. ll ans,d,b;
  62. int findst(int o,int l,int r) {
  63. if(l==r) return l;
  64. else {
  65. pushdown(o,l,r);
  66. int mid=l+r>>1,lc=o<<1,rc=lc|1;
  67. if(maxv[lc]+(d-maxt[lc])*A[mid]>b) return findst(lc,l,mid);
  68. return findst(rc,mid+1,r);
  69. }
  70. }
  71. void update(int o,int l,int r,int ql,int qr) {
  72. if(ql<=l&&r<=qr) {
  73. ans+=sumv[o]+d*suma[o]-b*(r-l+1);
  74. setv[o]=b;sett[o]=d;
  75. }
  76. else {
  77. pushdown(o,l,r);
  78. int mid=l+r>>1,lc=o<<1,rc=lc|1;
  79. if(ql<=mid) update(lc,l,mid,ql,qr);
  80. if(qr>mid) update(rc,mid+1,r,ql,qr);
  81. }
  82. maintain(o,l,r);
  83. }
  84. int main() {
  85. n=read();m=read();
  86. rep(i,1,n) A[i]=read();
  87. sort(A+1,A+n+1);
  88. build(1,1,n);
  89. rep(i,1,m) {
  90. d=read();b=read();
  91. if(maxv[1]+(d-maxt[1])*A[n]<=b) puts("0");
  92. else {
  93. int s=findst(1,1,n);ans=0;
  94. update(1,1,n,s,n);printf("%lld\n",ans);
  95. }
  96. }
  97. return 0;
  98. }

  

BZOJ4293: [PA2015]Siano的更多相关文章

  1. BZOJ4293 [PA2015]Siano(线段树)

    传送门 这Seg确实不好写,不过因为它与ai的相对顺序无关,所以,我们在对ai排序之后,就可做了.维护一个区间最大值,维护一个和,维护一个区间赋值的懒标记,再维护一个时间变化的标记就可以了. 因为不论 ...

  2. 【BZOJ4293】[PA2015]Siano 线段树

    [BZOJ4293][PA2015]Siano Description 农夫Byteasar买了一片n亩的土地,他要在这上面种草. 他在每一亩土地上都种植了一种独一无二的草,其中,第i亩土地的草每天会 ...

  3. 2018.07.23[PA2015]Siano(线段树)

    [PA2015]Siano 描述 Description 农夫Byteasar买了一片n亩的土地,他要在这上面种草. 他在每一亩土地上都种植了一种独一无二的草,其中,第i亩土地的草每天会长高a[i]厘 ...

  4. BZOJ.4293.[PA2015]Siano(线段树)

    题目链接 \(Description\) 有一片n亩的土地,要在这上面种草. 在每一亩土地上都种植了一种独一无二的草,其中,第\(i\)亩土地的草每天会长高\(a[i]\)厘米. 一共会进行\(m\) ...

  5. 【BZOJ】4293: [PA2015]Siano 线段树上二分

    [题意]给定n棵高度初始为0的草,每天每棵草会长高a[i],m次收割,每次在d[i]天将所有>b[i]的草收割到b[i],求每次收割量.n<=500000. [算法]线段树上二分 [题解] ...

  6. [PA2015]Siano 单调栈

    由于某人找了个单调栈的题解但是没研究透所以让我们来研究............ 首先先来考虑下面一种情况,假设第\(k\)次切割时,天数为\(d_k\),高度为\(b_k\),第\(k+1\)次切割时 ...

  7. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  8. 【bzoj4293】【PA2015】Siano

    如题,首先可以考虑相对大小是不变的. 那么先sort,之后每次在线段树上二分即可. #include<bits/stdc++.h> typedef long long ll; using ...

  9. BZOJ 4291: [PA2015]Kieszonkowe 水题

    4291: [PA2015]Kieszonkowe Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnli ...

随机推荐

  1. cd命令

    [cd]           切换目录     cd===>change directory 命令功能 : 切换目录到dirname 命令实例: 实例1:切换到跟目录下 命令: cd / 输出: ...

  2. Mysql 存储引擎中InnoDB与Myisam的主要区别

    一直以为我spring事物没有配置好,结果发现是mysql的表本身设置成了Myisam 引擎.改成innodb就支持事物了. 1, 事务处理 innodb 支持事务功能,myisam 不支持. Myi ...

  3. spring拦截器 实现应用之性能监控

    package cn.ximi.erp.web.common.interceptors; import cn.ximi.core.common.utils.string.StringUtil; imp ...

  4. DevExpress 隐藏Ribbon中barbuttonItem的SuperTip(1)

    public frmMain() { InitializeComponent(); ribbonControl1.Manager.HighlightedLinkChanged += Manager_H ...

  5. Excel中VBA进行插入列、格式化、排序

    在数据分析中经常需要对数据进行排序.排名,观察指标排名变化情况,手工处理的话不是太困难,但经常使用,还是编写宏比较方便. 宏命令比较简单,不多解释,只说一下注意事项: 1.有合并单元格,比如列.行合并 ...

  6. 移居 GitHub

    博客很久没能更新了,很多代码也从博客园逐渐转移到 GitHub,欢迎新老用户光顾: https://github.com/kedebug 个人博客:http://kedebug.me/

  7. Sublime 3 如何配置SVN插件

    在sublime里面安装svn的插件,就可以在sublime的操作界面里面进行相关svn操作,这样就不用再回到文件系统中,进行相关svn的操作. 1.在进入sublime界面后,点击顶部菜单“Pref ...

  8. winform中选择文件获取路径

    private void button1_Click(object sender, EventArgs e) { //此时弹出一个可以选择文件的窗体 OpenFileDialog fileDialog ...

  9. MFC 静态文本的自绘 空心字的实现

    想在对话框里,显示几个字是很简单的,只要用静态文本的输出就可以了.然而有时候我们需要显示特效的字,我们希望显示的文字就像Word里的艺术字一样,看起来美观.这时我们可以重写CStatic类.用Draw ...

  10. 鼠标右击.exe的程序出现闪退(桌面重启)怎么办

    桌面鼠标右击,然后点NVIDIA控制面板(或者按windows+S组合键,在底下输入NVIDIA,然后点NVIDIA控制面板)   好了,然后任意找个文件试试...如图,问题完美解决