#1034 : 毁灭者问题

时间限制:10000ms
单点时限:1000ms
内存限制:256MB

描述

在 Warcraft III 之冰封王座中,毁灭者是不死族打三本后期时的一个魔法飞行单位。

毁灭者的核心技能之一,叫做魔法吸收(Absorb Mana):

现在让我们来考虑下面的问题:

假设你拥有 n 个魔法单位,他们从左到有站在一行,编号从 1 到 n。 每个单位拥有三项属性:

  • si: 初始法力。

  • mi: 最大法力上限。

  • ri: 每秒中法力回复速度。

现在你操纵一个毁灭者,有 m 个操作,t l r,表示时刻 t,毁灭者对所有编号从 l 到 r 的单位,使用了魔法吸收。操作按照时间顺序给出,计算毁灭者一共吸收了多少法力。

输入

输入数据的第一行有一个整数 n(1 ≤  n ≤105) — 你的魔法单位的数目。

接下来的 n 行,每行有三个整数 si, mi, ri(0 ≤ si ≤ mi ≤ 105, 0 ≤ ri ≤ 105) 描述一个魔法单位。

接下来一行又一个整数 m(1 ≤ m ≤ 105), — 操作的数目。

接下来的 m 行,每行描述一个操作 t, l, r(0 ≤ t ≤ 109, 1 ≤ l ≤ r ≤ n),t 非降。

输出

输出一行一个整数表示毁灭者一共吸收了多少法力。

样例输入
  1. 5
  2. 0 10 1
  3. 0 12 1
  4. 0 20 1
  5. 0 12 1
  6. 0 10 1
  7. 2
  8. 5 1 5
  9. 19 1 5
样例输出
    83
题解:
    自己写了一棵线段树的做法,看了网上一篇辣鸡博客,哎,无语了,
    自己写的时候是知道的这个写法比暴力还暴力,n^2logn的,但是还是
    写了,结果TLE,T飞了。
附上T飞代码
  1. #include<cstring>
  2. #include<cmath>
  3. #include<algorithm>
  4. #include<iostream>
  5. #include<cstdio>
  6.  
  7. #define N 100007
  8. #define ll long long
  9. using namespace std;
  10. inline int read()
  11. {
  12. int x=,f=;char ch=getchar();
  13. while(ch<''||ch>''){if (ch=='-')f=-;ch=getchar();}
  14. while(ch>=''&&ch<=''){x=(x<<)+(x<<)+ch-'';ch=getchar();}
  15. return x*f;
  16. }
  17.  
  18. int n,m;
  19. ll ans;
  20. struct Node
  21. {
  22. ll t,m,r,s;
  23. }tr[N*];
  24.  
  25. inline void update(int p)
  26. {
  27. tr[p].s=tr[p<<].s+tr[p<<|].s;
  28. }
  29. void build(int p,int l,int r)
  30. {
  31. if (l==r)
  32. {
  33. tr[p].s=read(),tr[p].m=read(),tr[p].r=read();
  34. tr[p].t=;
  35. return;
  36. }
  37. int mid=(l+r)>>;
  38. build(p<<,l,mid),build(p<<|,mid+,r);
  39. update(p);
  40. }
  41. void renew(int p,int l,int r,int x,int y,int t)
  42. {
  43. if (l==r)
  44. {
  45. if (tr[p].s+(t-tr[p].t)*tr[p].r>tr[p].m) tr[p].s=tr[p].m;
  46. else tr[p].s+=(t-tr[p].t)*tr[p].r;
  47. tr[p].t=t;
  48. return;
  49. }
  50. int mid=(l+r)>>;
  51. if (y<=mid) renew(p<<,l,mid,x,y,t);
  52. else if (x>mid) renew(p<<|,mid+,r,x,y,t);
  53. else renew(p<<,l,mid,x,mid,t),renew(p<<|,mid+,r,mid+,y,t);
  54. update(p);
  55. }
  56. ll query(int p,int l,int r,int x,int y)
  57. {
  58. if (l==x&&y==r) return tr[p].s;
  59. int mid=(l+r)>>;
  60. if (y<=mid) return query(p<<,l,mid,x,y);
  61. else if (x>mid) return query(p<<|,mid+,r,x,y);
  62. else return query(p<<,l,mid,x,mid)+query(p<<|,mid+,r,mid+,y);
  63. }
  64. void clean_all(int p,int l,int r,int x,int y)
  65. {
  66. if (l==r)
  67. {
  68. tr[p].s=;
  69. return;
  70. }
  71. int mid=(l+r)>>;
  72. if (y<=mid) return clean_all(p<<,l,mid,x,y);
  73. else if (x>mid) return clean_all(p<<|,mid+,r,x,y);
  74. else clean_all(p<<,l,mid,x,mid),clean_all(p<<|,mid+,r,mid+,y);
  75. update(p);
  76. }
  77. int main()
  78. {
  79. freopen("fzy.in","r",stdin);
  80. freopen("fzy.out","w",stdout);
  81.  
  82. n=read(),build(,,n);
  83. m=read();
  84. for (int i=;i<=m;i++)
  85. {
  86. int t=read(),x=read(),y=read();
  87. renew(,,n,x,y,t);
  88. ans+=query(,,n,x,y);
  89. clean_all(,,n,x,y);
  90. }
  91. printf("%lld",ans);
  92. }

    正解思路

    对于s,m,r我们可以这样想,

    对于输入的间隔tk-tk-1,设为d,如果d*r>m 则为m 1

                   如果d*r<=m,则为r*d    2

    所以答案就等于所以人,满足1的个数乘以m,以及满足2的∑di  *r,这个就是答案。

    

    我们应该对于每个人来计算答案,对于每个人,如果暴力计算的话,就是O(nm)对吧。

    还是T飞,那怎么办呢,可以用一棵平衡树+线段树来维护,对于每个起始时间,结束时间

    都放入平衡树中,间隔即为判断条件,可以放入线段树中,结束时间时在维护,一下,删除

    所以,每个询问只会被插入一次,删除一次,对于每个人询问一次,所以总复杂度为O(nlgn)。

    代码,莫名wrong,但是对拍没有错,就精神ac吧。

  1. #include<cstring>
  2. #include<cmath>
  3. #include<algorithm>
  4. #include<cstdio>
  5. #include<iostream>
  6. #include<set>
  7. #include<vector>
  8.  
  9. #define lson tr[p].ls
  10. #define rson tr[p].rs
  11. #define z1 set<int>::iterator
  12. #define N 100007
  13. #define ll long long
  14. using namespace std;
  15. inline int read()
  16. {
  17. int x=,f=;char ch=getchar();
  18. while(ch<''||ch>''){if (ch=='-')f=-;ch=getchar();}
  19. while(ch>=''&&ch<=''){x=(x<<)+(x<<)+ch-'';ch=getchar();}
  20. return x*f;
  21. }
  22.  
  23. int n,m,root,sz;
  24. ll ans;
  25. vector<int>hd[N],ed[N];
  26. set<int>q;//哪几个时间段有
  27. struct Node
  28. {
  29. ll s,m,r;
  30. }a[N];
  31. struct Date
  32. {
  33. int num,sum,ls,rs;
  34. }tr[];
  35.  
  36. inline void update(int p)
  37. {
  38. tr[p].sum=tr[lson].sum+tr[rson].sum;
  39. tr[p].num=tr[lson].num+tr[rson].num;
  40. }
  41. void add(int &p,int l,int r,int x,int flag)
  42. {
  43. if(!p) p=++sz;
  44. if (l==r)
  45. {
  46. tr[p].sum+=x*flag,tr[p].num+=flag;
  47. return;
  48. }
  49. int mid=(l+r)>>;
  50. if (x<=mid) add(lson,l,mid,x,flag);
  51. else add(rson,mid+,r,x,flag);
  52. update(p);
  53. }
  54. ll query1(int p,int l,int r,int x,int y)
  55. {
  56. if (!p) return ;
  57. if (l==x&&r==y) return tr[p].sum;
  58. int mid=(l+r)>>;
  59. if (y<=mid) return query1(lson,l,mid,x,y);
  60. else if (x>mid) return query1(rson,mid+,r,x,y);
  61. else return query1(lson,l,mid,x,mid)+query1(rson,mid+,r,mid+,y);
  62. }
  63. ll query2(int p,int l,int r,int x,int y)
  64. {
  65. if (!p) return ;
  66. if (l==x&&r==y) return tr[p].num;
  67. int mid=(l+r)>>;
  68. if (y<=mid) return query2(lson,l,mid,x,y);
  69. else if (x>mid) return query2(rson,mid+,r,x,y);
  70. else return query2(lson,l,mid,x,mid)+query2(rson,mid+,r,mid+,y);
  71. }
  72. int main()
  73. {
  74. freopen("fzy.in","r",stdin);
  75. freopen("solution.out","w",stdout);
  76.  
  77. n=read();
  78. for (int i=;i<=n;i++)
  79. a[i].s=read(),a[i].m=read(),a[i].r=read();
  80. m=read();
  81. for (int i=;i<=m;i++)
  82. {
  83. int t=read(),x=read(),y=read();
  84. hd[x].push_back(t);
  85. ed[y].push_back(t);
  86. }
  87. q.insert(-),q.insert(1e9+);//放一个哨兵。
  88.  
  89. for (int i=;i<=n;i++)
  90. {
  91. for (int j=;j<hd[i].size();j++)
  92. {
  93. z1 qq=q.lower_bound(hd[i][j]),hj=q.upper_bound(hd[i][j]);qq--;
  94. if (*qq==-&&*hj==1e9+) q.insert(hd[i][j]);
  95. else if (*qq==-)
  96. {
  97. q.insert(hd[i][j]);
  98. add(root,,,*hj-hd[i][j],);
  99. }
  100. else if (*hj==1e9+)
  101. {
  102. q.insert(hd[i][j]);
  103. add(root,,,hd[i][j]-*qq,);
  104. //cout<<hd[i][j]-*qq<<" flag"<<endl;
  105. }
  106. else
  107. {
  108. add(root,,,*hj-*qq,-);
  109. q.insert(hd[i][j]);
  110. add(root,,,hd[i][j]-*qq,),add(root,,,*hj-hd[i][j],);
  111. }
  112. }
  113. int up=ceil(a[i].m*1.0/(double)a[i].r);
  114. ans+=query1(root,,,,up-)*a[i].r;
  115. ans+=query2(root,,,up,)*a[i].m;
  116. z1 t=q.begin();t++;
  117. if (*t!=)
  118. {
  119. if (*t*a[i].r+a[i].s>a[i].m) ans+=a[i].m;
  120. else ans+=*t*a[i].r+a[i].s;
  121. }
  122. for (int j=;j<ed[i].size();j++)
  123. {
  124. z1 qq=q.lower_bound(ed[i][j]),hj=q.upper_bound(ed[i][j]);qq--;
  125. if (*qq==-&&*hj==1e9+) q.erase(ed[i][j]);
  126. else if (*qq==-)
  127. {
  128. q.erase(ed[i][j]);
  129. add(root,,,*hj-ed[i][j],-);
  130. }
  131. else if (*hj==1e9+)
  132. {
  133. q.erase(ed[i][j]);
  134. add(root,,,ed[i][j]-*qq,-);
  135. }
  136. else
  137. {
  138. add(root,,,ed[i][j]-*qq,-),add(root,,,*hj-ed[i][j],-);
  139. q.erase(ed[i][j]);
  140. add(root,,,*hj-*qq,);
  141. }
  142. }
  143. // cout<<ans<<endl;
  144. }
  145. printf("%lld",ans);
  146. }

hihocoder #1034 : 毁灭者问题 平衡树(set)+线段树的更多相关文章

  1. 【学习笔记】浅析平衡树套线段树 & 带插入区间K小值

    常见的树套树 一般来说,在嵌套数据结构中,线段树多被作为外层结构使用. 但线段树毕竟是 静态 的结构,导致了一些不便. 下面是一个难以维护的例子: 带插入区间 \(k\) 小值问题 来源:Luogu ...

  2. hihocoder #1034 毁灭者问题

    传送门 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 在 Warcraft III 之冰封王座中,毁灭者是不死族打三本后期时的一个魔法飞行单位. 毁灭者的核心技能之一, ...

  3. bzoj 3196 Tyvj 1730 二逼平衡树(线段树套名次树)

    3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1807  Solved: 772[Submit][Stat ...

  4. BZOJ3196 二逼平衡树 ZKW线段树套vector(滑稽)

    我实在是不想再打一遍树状数组套替罪羊树了... 然后在普通平衡树瞎逛的时候找到了以前看过vector题解 于是我想:为啥不把平衡树换成vector呢??? 然后我又去学了一下ZKW线段树 就用ZKW线 ...

  5. bzoj 3196/ Tyvj 1730 二逼平衡树 (线段树套平衡树)

    3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description ...

  6. [bzoj3065] 带插入区间第k小值 [重量平衡树套线段树]

    题面 传送门 思路 发现强制在线了...... 本来可以树套树解决的问题,现在外层不能使用线段树了,拿什么替代呢? 我们需要一种支持单点插入.下套数据结构.数据结构上传合并复杂度最多单log,不能旋转 ...

  7. BZOJ3196 二逼平衡树 【线段树套平衡树】

    题目 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查询k在区间内的前驱(前驱 ...

  8. CEOI 2019 Day2 T2 魔法树 Magic Tree (LOJ#3166、CF1993B、and JOI2021 3.20 T3) (启发式合并平衡树,线段树合并)

    前言 已经是第三次遇到原题. 第一次是在 J O I 2021 S p r i n g C a m p \rm JOI2021~Spring~Camp JOI2021 Spring Camp 里遇到的 ...

  9. [BZOJ 3196] 213平衡树 【线段树套set + 树状数组套线段树】

    题目链接:BZOJ - 3196 题目分析 区间Kth和区间Rank用树状数组套线段树实现,区间前驱后继用线段树套set实现. 为了节省空间,需要离线,先离散化,这样需要的数组大小可以小一些,可以卡过 ...

随机推荐

  1. GC_CONCURRENT freed 循环不停打印日志

    打印类似如下语句: 03-07 19:21:49.562: D/dalvikvm(1677): GC_CONCURRENT freed 2859K, 20% free 12020K/15011K, p ...

  2. SpringCloud开发学习总结(五)—— 服务容错保护Hystrix

    在微服务架构中,我们将系统拆分成了很多服务单元,各单元的应用间通过服务注册与订阅的方式相互依赖.但由于每个单元都在不同的进程中运行,一来通过远程调用的方式执行,这样就有可能因为网络原因或是依赖服务自身 ...

  3. Vmware workstation12里如何正确快速安装可视化IDS系统Security Onion(图文详解)

    不多说,直接上干货! 首先,大家要明确: 问:安全洋葱能阻止入侵吗? 答:这一点,和OSSIM一样,不能阻止入侵. Security Onion基于Ubuntu,包含了入侵检测.网络安全监控.日志管理 ...

  4. SpringSecurity的简单使用

    导入SpringSecurity坐标 在web.xml中配置过滤器 编写spring-securiy配置文件 编写自定义认证提供者 用户新增时加密密码 配置页面的login和logout 获取登录用户 ...

  5. 建造者模式以及php实现

    建造者模式: 造者模式(Builder Pattern):将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 建造者模式是一步一步创建一个复杂的对象,它允许用户只通过指定复杂对 ...

  6. Java用SAX解析XML

    要解析的XML文件:myClass.xml <?xml version="1.0" encoding="utf-8"?> <class> ...

  7. 从0开始搭建SQL Server 2012 AlwaysOn 第二篇(配置故障转移集群)

    本篇主要讲配置Windows 故障转移集群及遇到的相关问题(坑),因为AlwaysOn是基于Windows的故障转移集群的 在讲解步骤之前需要了解一下故障转移集群仲裁配置 四种集群的仲裁配置: 1.多 ...

  8. jQuery之Validation表单验证插件使用

    <!DOCTYPE html><html xmlns="http://www.w3.org/1999/xhtml"><head> <met ...

  9. 恩智浦Freescale Cortex-A9 迅为IMX6开发板平台初体验

    iTOP-i.MX6 开发板预装 Android4.4 系统,采用 9.7 寸(或者 7 寸或者 4.3 寸)IPS 屏 幕,至少 5 点以上触控,操作流畅,无论是高清视频.游戏等都会有上佳的表现,实 ...

  10. nginx配置实现负载均衡

    一.负载均衡的作用 1.转发功能 按照一定的算法[权重.轮询],将客户端请求转发到不同应用服务器上,减轻单个服务器压力,提高系统并发量. 2.故障移除 通过心跳检测的方式,判断应用服务器当前是否可以正 ...