2018 Multi-University Training Contest 2-1007(hdu 6315)-题解
一、题意
给定一个元素个数为$N(1 \le N \le 10^5)$初始序列$a$和$b$,$a$序列的初始值全为$0$,$b$序列的初始值为$1$到$N$的一个排列。有$T(1 \le T \le 10^5)$次操作。操作有如下两种类型:
1、$add\ l\ r$:给序列$a$的区间$[l, r]$内所有元素加$1$;
2、$query\ l\ r$:查询$\sum\limits_{i=l}^{r}\lfloor{\frac{a_i}{b_i}}\rfloor$,并输出。
二、简要思路概括
因为所有操作的总和$tot \le \sum\limits_{i=1}^{N}\frac{T}{i} = T * \sum\limits_{i=1}^{N}\frac{1}{i}$,而可证明调和级数$T * \sum\limits_{i=1}^{N}\frac{1}{i} \le TlogN$。
所以,建立两棵线段树。一棵树$B$维护$b$序列的区间最小值,可支持区间修改。另一棵树$C$维护$\lfloor{\frac{a_i}{b_i}}\rfloor$的区间和。
每次遇到$add$操作时,将$B$线段树的区间$[l, r]$减$1$,时间复杂度$O(logN)$。然后,再对线段树$B$做一遍dfs,如果有叶子节点的值为$0$,则把线段树$C$的对应位置加$1$,同时将该位置的值设置为初始的$b$值。时间复杂度$O(N)$。
所以,单次$add$操作的时间复杂度是$O(N)$,总的时间复杂度是$O(N*T)$。但是,不要觉得会超时。实际的计算量并没有这么大。因为前面已证明,最多只会累加$T \times logN$次。所以,只要代码写得优雅,绝对不会超时。另外,还有一种写法,就是每次对线段树$B$做区间减$1$之后,循环查询$B$线段树的$[l, r]$区间内的最小值,如果最小值为$0$,则把线段树$C$的对应位置加$1$,同时修改该位置的值为初始的$b$值,直到该区间内的最小值不为$0$为止。这样,总的时间复杂度是$O(T*NlogN)$,也不会超时。
三、代码
#include<bits/stdc++.h> using namespace std; #define MAXN 100010 typedef long long ll; struct Node1 { int mv, lazy; }; struct Node2 { int v, sum; }; Node1 b[MAXN * ]; Node2 c[MAXN * ]; int b0[MAXN], N, Q; void push_down(int rt) { if(b[rt].lazy) { , rch = rt << | ; b[lch].mv += b[rt].lazy, b[lch].lazy += b[rt].lazy; b[rch].mv += b[rt].lazy, b[rch].lazy += b[rt].lazy; b[rt].mv = min(b[rt << ].mv, b[rt << | ].mv); b[rt].lazy = ; } } , , int r = N) { if(l >= ul && r <= ur) { if(f)b[rt].mv = val; else b[rt].mv += val, b[rt].lazy += val; return; } push_down(rt); ; , l, mid); | , mid + , r); b[rt].mv = min(b[rt << ].mv, b[rt << | ].mv); } , , int r = N) { if(l == r && l == p) { c[rt].v++, c[rt].sum++; return; } ; , l, mid); | , mid + , r); c[rt].sum = c[rt << ].sum + c[rt << | ].sum; } , , int r = N) { if(ql <= l && qr >= r)return c[rt].sum; , res = ; , l, mid); | , mid + , r); return res; } , , int r = N) { if(l == r) { b[rt].mv = b0[l], update2(l); return; } push_down(rt); ; ].mv == )dfs(ul, ur, rt << , l, mid); | ].mv == )dfs(ul, ur, rt << | , mid + , r); b[rt].mv = min(b[rt << ].mv, b[rt << | ].mv); } int main() { // freopen("input.txt", "r", stdin); ]; int l, r; while(~scanf("%d%d", &N, &Q)) { , tt = N << ; i < tt; ++i)b[i].mv = INT_MAX, b[i].lazy = ; memset(c, , sizeof(c)); ; i <= N; ++i)scanf(); while(Q--) { scanf("%s%d%d", cmd, &l, &r); ] == 'a') { update1(l, r, -, ); dfs(l, r); } else cout << query2(l, r) << endl; } } ; }
2018 Multi-University Training Contest 2-1007(hdu 6315)-题解的更多相关文章
- hdu 4941 2014 Multi-University Training Contest 7 1007
Magical Forest Time Limit: 24000/12000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Other ...
- 2017 Multi-University Training Contest - Team 2 &&hdu 6050 Funny Function
Funny Function Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...
- 2017 Multi-University Training Contest - Team 2 &&hdu 6053 TrickGCD
TrickGCD Time Limit: 5000/2500 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total ...
- 2017 Multi-University Training Contest - Team 2&&hdu 6047 Maximum Sequence
Maximum Sequence Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- hdu 6315 Naive Operations (2018 Multi-University Training Contest 2 1007)
Naive Operations Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 502768/502768 K (Java/Other ...
- HDU 6356.Glad You Came-线段树(区间更新+剪枝) (2018 Multi-University Training Contest 5 1007)
6356.Glad You Came 题意就是给你一个随机生成函数,然后从随机函数里确定查询的左右区间以及要更新的val值.然后最后求一下异或和就可以了. 线段树,区间最大值和最小值维护一下,因为数据 ...
- HDU 6315.Naive Operations-线段树(两棵树合并)(区间单点更新、区间最值、区间求和)+思维 (2018 Multi-University Training Contest 2 1007)
6315.Naive Operations 题意很好理解,但是因为区间求和求的是向下取整的a[i]/b[i],所以直接分数更新区间是不对的,所以反过来直接当a[i]==b[i]的时候,线段树对应的位置 ...
- 【2018 Multi-University Training Contest 2 1007】Naive Operations
[链接] 我是链接,点我呀:) [题意] 给你两个数组a,b; b数组是1..n的一个排列. 现在给你两种操作: add l,r将a[l..r]都加上1 query l,r 询问$∑^r_l\frac ...
- 2018 Nowcoder Multi-University Training Contest 2
目录 Contest Info Solutions A. run D. monrey G. transform H. travel I. car J. farm Contest Info Practi ...
随机推荐
- 在 if 条件句中使用向量
上述例子的函数中,都只输入了单一值.如果我们提供一个向量,那么这个函数将产生警告,这是因为 if 语句不能与多值向量共存.check_ _positive(c(1,-1,0))## Warning i ...
- 【Docker】Windows下docker环境搭建及解决使用非官方终端时的连接问题
背景 时常有容器方面的需求,经常构建调试导致测试环境有些混乱,所以想在本地构建一套环境,镜像调试稳定后再放到测试环境中. Windows Docker 安装 安装docker toolbox 博主的电 ...
- javascript之分时函数
在一些开发场景中,我们可能会一次性向文档中注入上千个节点,在短时间内向浏览器中大量添加DOM节点可能会让浏览器吃不消,结果往往会让浏览器卡顿或吃不消,解决方案之一便是使用分时函数(timeChunk) ...
- zoj3261变形并查集
需要变形的并查集,这题错了好久,一直没a掉,终于在重写第三次的时候a了 先保存数据,把不需要拆分的边合并,逆向计算,需要拆分时就合并,之前不知道为啥写搓了,tle好久 #include<map& ...
- @RequestParam注解的作用及用法
最简单的两种写法,在写接口时:加或不加@RequestParam注解的区别 第一种写法参数为非必传,第二种写法参数为必传.参数名为userId 第二种写法可以通过@RequestParam(requi ...
- VGG16提取图像特征 (torch7)
VGG16提取图像特征 (torch7) VGG16 loadcaffe torch7 下载pretrained model,保存到当前目录下 th> caffemodel_url = 'htt ...
- HDU 1969 精度二分
Pie Time Limit: 5000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...
- dbms_job.submit方式创建job,太老了
--方法一declarejobno number; begin dbms_job.submit(jobno, 'xxxx;', xxxx, 'xxxx'); commit ...
- 10046event vs 10053event做了哪些操作
10046 event 告诉我们oracle做了哪些事情,做的怎么样.10053 event 告诉我们oracle优化器怎么想的,再给出成本最低的最终的优化方案,考虑了哪些执行计划拿了哪些统计信息
- 【转】MongoDB 3.0 正式版本即将发布,强力推荐
MongoDB 今天宣布3.0 正式版本即将发布.这标志着 MongoDB 数据库进入了一个全新的发展阶段,提供强大.灵活而且易于管理的数据库管理系统. MongoDB 3.0 在性能和伸缩性方面都有 ...