Q - Play With Sequence HDU - 3971 线段树 重新排序建树
Q - Play With Sequence
这个题目是一个线段树,比较特别的线段树,就是c询问一定次数之后重新排序建树来优化减低复杂度。
第一次碰到这种题目有点迷。
这个题目写还是很好写的,就是重新排序建树的位置不太好找。
不过可以知道的是,这是更新花费时间和排序花费时间的一个平衡,这个是一个二次函数,这个二次函数的最低点可以自己测出来。
现在可能有点听不懂,写完代码就很好理解了,
我测的每隔2000次C的操作就重新建树排序是最优的。
800,1000,2200,3000 都是可以的。
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm>
#include <iostream>
#include <queue>
#include <string>
#include <vector>
#include <map>
#define inf 0x3f3f3f3f
#define inf64 0x3f3f3f3f3f3f3f3f
using namespace std;
const int maxn = 3e5 + ;
typedef long long ll;
ll a[maxn], lc[maxn], rc[maxn], num[maxn];
string s[maxn];
struct node
{
ll lazy, max, min, len;
}tree[maxn*];
void push_up(int id)
{
tree[id].max = max(tree[id << ].max, tree[id << | ].max);
tree[id].min = min(tree[id << ].min, tree[id << | ].min);
//printf("tree[%d].min=%lld tree[%d].max=%lld\n", id, tree[id].min, id, tree[id].max);
} void build(int id,int l,int r)
{
tree[id].len = r - l + ;
tree[id].lazy = ;
if(l==r)
{
tree[id].max = tree[id].min = a[l];
return;
}
int mid = (l + r) >> ;
build(id << , l, mid);
build(id << | , mid + , r);
push_up(id);
} void push_down(int id)
{
if(tree[id].lazy)
{
int val = tree[id].lazy;
tree[id << ].max += val;
tree[id << ].min += val;
tree[id << | ].max += val;
tree[id << | ].min += val;
tree[id << ].lazy += val;
tree[id << | ].lazy += val;
// printf("tree[%d].max=%lld tree[%d].min=%lld\n", id << 1, tree[id << 1].max, id << 1, tree[id << 1].min);
// printf("tree[%d].max=%lld tree[%d].min=%lld\n", id << 1 | 1, tree[id << 1 | 1].max, id << 1 | 1, tree[id << 1 | 1].min);
tree[id].lazy = ;
}
}
void update(int id,int l,int r,ll x,ll y,ll val)
{
push_down(id);
// printf("id=%d l=%d r=%d x=%lld y=%lld val=%lld\n", id, l, r, x, y, val);
if(tree[id].min>=x&&tree[id].max<=y)
{
tree[id].lazy = val;
tree[id].min += val;
tree[id].max += val;
//printf("id=%d min=%lld max=%lld\n", id, tree[id].min, tree[id].max);
return;
}
int mid = (l + r) >> ;
if (tree[id << ].max >= x && tree[id << ].min <= y) update(id << , l, mid, x, y, val);
if (tree[id << | ].max >= x && tree[id << | ].min <= y) update(id << | , mid + , r, x, y, val);
push_up(id);
} int query(int id,int l,int r,ll x,ll y)
{
push_down(id);
if(tree[id].max<=y&&tree[id].min>=x)
{
return tree[id].len;
}
int mid = (l + r) >> , ans = ;
if (tree[id << ].max >= x && tree[id << ].min <= y) ans += query(id << , l, mid, x, y);
if (tree[id << | ].max >= x && tree[id << | ].min <= y) ans += query(id << | , mid + , r, x, y);
return ans;
} void push_alldown(int id,int l,int r)
{
if(l==r)
{
a[l] = tree[id].max;
return;
}
push_down(id);
int mid = (l + r) >> ;
push_alldown(id << , l, mid);
push_alldown(id << | , mid + , r);
} int main()
{
int n, m;
while(scanf("%d%d",&n,&m)!=EOF)
{
for (int i = ; i <= n; i++) scanf("%lld", &a[i]);
sort(a + , a + + n);
build(, , n);
int cnt = ;
for(int i=;i<=m;i++)
{
cin >> s[i];
if (s[i] == "C") scanf("%lld%lld%lld", &lc[i], &rc[i], &num[i]), cnt++;
else scanf("%lld%lld", &lc[i], &rc[i]), num[i] = ;
}
int tot = ;
for(int i=;i<=m;i++)
{
if(s[i]=="C")
{
++tot;
//push_alldown(1, 1, n);
//printf("lc[%d]=%lld rc[%d]=%lld num[%d]=%lld\n", i, lc[i], i, rc[i], i, num[i]);
update(, , n, lc[i], rc[i], num[i]);
if(tot%==)
{
push_alldown(, , n);
sort(a + , a + + n);
build(, , n);
}
}
else
{
int ans = query(, , n, lc[i], rc[i]);
printf("%d\n", ans);
}
}
}
}
线段树 排序建树
Q - Play With Sequence HDU - 3971 线段树 重新排序建树的更多相关文章
- hdu 3436 线段树 一顿操作
Queue-jumpers Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- hdu 3397 线段树双标记
Sequence operation Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- hdu 2871 线段树(各种操作)
Memory Control Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...
- hdu 4267 线段树间隔更新
A Simple Problem with Integers Time Limit: 5000/1500 MS (Java/Others) Memory Limit: 32768/32768 K ...
- hdu 4747 线段树
Mex Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Submis ...
- hdu 3954 线段树 (标记)
Level up Time Limit: 10000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- hdu 1754 线段树(Max+单点修改)
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- hdu 1166 线段树(sum+单点修改)
敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submi ...
- hdu 5877 线段树(2016 ACM/ICPC Asia Regional Dalian Online)
Weak Pair Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total ...
随机推荐
- 几行代码实现cookie的盗取
前言 上一篇文章中介绍了XSS(跨站脚本攻击)简单原理与几种类型.接下来通过实例用几行代码实现cookie的盗取. 正文 这里测试用的工具是DVWA(可以本地搭建,前面文章有介绍),和phpstudy ...
- LIMS产品 - Labvantage技术版本
最新版本的Labvantage8使用Java版本为Java7(Java8使用最广泛,最新版本为Java12),中间件使用JBoss(国内小型信息化系统(LIMS.QMS等)java体系一般使用Tomc ...
- threejs点击事件
示例浏览地址:https://ithanmang.gitee.io/threejs/home/201807/20180703/02-raycasterDemo.html 双击鼠标左键选中模型并显示信息 ...
- Web三维编程入门总结之二:面向对象的基础Web3D框架
本篇主要通过分析Tony Parisi的sim.js库(原版代码托管于:https://github.com/tparisi/WebGLBook/tree/master/sim),总结基础Web3D框 ...
- 如何用python批量生成真实的手机号码
前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者:Python测试社区 1目 标 场 景 平时在工作过程中,偶尔会需要大 ...
- 记一次错误 POST http://127.0.0.1:8000/auth/signup/ 500 (Internal Server Error)
错误环境: 1,ajax 发送 post 请求调用 views 的视图 xfzajax.post({ 'url': '/xfzauth/signup/', 'data': { 'telephone': ...
- jdbctemplate打印sql
在logback.xml里加入如下配置即可: <include resource="org/springframework/boot/logging/logback/base.xml& ...
- Youtube推荐算法的前世今生
第一阶段,基于User-Video图游历算法,2008年[1]. 在这个阶段,YouTube认为应该给用户推荐曾经观看过视频的同类视频,或者说拥有同一标签的视频.然而此时,YouTube的视频已是数千 ...
- Certified Scrum Master CSM 中文资料大全
课程概览 本课程由中国唯一一位获CST认证培训师及LeSS-Friendly Scrum Trainer双重认证讲师,丰富一线实战经验的Scrum教练讲授:姜信宝 BoB Jiang. 敏捷变革中心是 ...
- tp5 -- 控制器的参数
方法的参数是可以直接获取的到get和post这集中提交格式的数据的. 但是呢. 前置操作时不能这样操作的, 只能老老实实的使用input()这个方法来获取!!!