2018 杭电多校2 - Naive Operations
b is a static permutation of 1 to n. Initially a is filled with zeroes.
There are two kind of operations:
1.add l r: add one for $$$a_l, a_{l+1}...a_r$$$
2.query l r: query $$$\sum_{i=l}^{r}\lfloor a_i/b_i\rfloor$$$
For each test case, in the first line, two integers n,q, representing the length of a,b and the number of queries.
In the second line, n integers separated by spaces, representing permutation b.
In the following q lines, each line is either in the form 'add l r' or 'query l r', representing an operation.
$$$1\le n,q\le 100000, 1\le l\le r\le n$$$, there're no more than 5 test cases.
1 5 2 4 3
add 1 4
query 1 4
add 2 5
query 2 5
add 3 5
query 1 5
add 2 4
query 1 4
add 2 5
query 2 5
add 2 2
query 1 5
1
2
4
4
6
看到add l r和query l r,就觉得这是一道线段树,可是$$$b_i$$$的存在又破坏了区间的统一性。
根据线段树的特点,为了把add和query的复杂度降低到log(n),结点应该记录这个区间被add了几次,以及区间的和$$$\sum_{i=l}^{r}\lfloor a_i/b_i\rfloor$$$,核心问题在于怎样更新每个结点的和。
由于复杂度的原因,不能每次都重新算一遍所有结点的和,但因为每次add操作都只加1,很多结点的值都不会立刻变化;实际上区间的和的变化有这样的特点:在若干次add之前,区间和都不会改变,只有当某一项$$$a_i$$$增加到$$$b_i$$$的倍数时,区间和才会相应的+1,也就是说,在整个区间被加若干次之前,都不用更新它,只把它总共被加的次数+1;一旦某一项变成$$$a_i$$$增加到$$$b_i$$$的倍数时,区间和+1,可以认为$$$a_i$$$又恢复为0,接下来整个区间又需要若干次才会改变。如果在区间和未改变时省掉更新操作,那么复杂度将一定程度上降低。
PS:题解视频里也讲啦,总更新次数很少的
考虑这样一个标签,它记录的是,整个区间至少被加几次以后,才会让其中一项发生变化。最开始的时候,$$$a_i$$$的标签都是$$$a_i$$$,父亲结点则记录两个儿子的标签较小的那个,当标签的值变为0的时候,更新区间和,并把对应的结点的标签重新设为$$$a_i$$$,并更新路径上的所有标签;而在标签大于0的时候,则不需要对区间进行更新。
具体的来说,假设用sum[]记录区间被完整的加了几次,low[]记录区间再被加几次就需要更新,ans[]记录区间和,每次add操作,找到对应的区间i,把sum[i]++,low[i]--,如果low[i]等于0了,就向下更新,把子区间的sum加上sum[i],low减去sum[i],把所有low变为小于等于0的区间继续向下更新,直到把发生变化的那个$$$a_i$$$重设为0,ans[]++,然后在回溯的时候,更新路径上的所有low[]和ans[]。
线段树写的有点烂。。。勉强过了
#include<stdio.h>
#include <cstring>
#define N_max 100005
typedef long long LL;
#define min(a,b) ((a)<(b)?(a):(b))
int lg[N_max << ], rg[N_max << ]
, low[N_max << ]//还要加几次
, sum[N_max << ]//已经加几次
, b[N_max << ]//bi
,ans[N_max<<]//区间的和
; #define lid(x) (x<<1)
#define rid(x) (x<<1|1)
void build(int id,int cl,int cr)
{
lg[id] = cl;
rg[id] = cr;
if (cl == cr)//build的时候完成输入,并初始化low为bi
{
scanf("%d", b + id);
low[id] = b[id]; return;
}
int left = lid(id),right=rid(id),cm= (cl + cr) >> ;
build(left, cl, cm);
build(right, cm + , cr);
low[id] = min(low[left], low[right]);//结点记录最小的low
} //low为0时,向下更新,直接处理整个区间
void addup(int id,int v)
{
sum[id] += v;
low[id] -= v;
if(lg[id]==rg[id])//遇到叶子结点
{
ans[id] += (sum[id]/ b[id]);
low[id] =b[id]- (sum[id] % b[id]);
sum[id] %=b[id] ;
}
if(low[id]<=)//low<=0说明区间内的某个数发生变化,需要继续向下更新
{
addup(lid(id), sum[id]);
addup(rid(id), sum[id]);
sum[id] = ;
ans[id] = ans[lid(id)] + ans[rid(id)];
low[id]=min(low[lid(id)], low[rid(id)]);
}
} inline void pushdown(int id)
{
if (sum[id])
{
sum[lid(id)] += sum[id];
low[lid(id)] -= sum[id];
sum[rid(id)] += sum[id];
low[rid(id)] -= sum[id];
sum[id] = ;
}
}
void add(int id,int cl,int cr,int v)
{
if (lg[id] == rg[id])//叶子结点
{
low[id]--; sum[id]++;//当进入叶子结点时,sum的值其实就是ai
ans[id] += (sum[id] / b[id]);//ai中有几个bi,ans就应该加几
low[id] = b[id] - (sum[id] % b[id]);//low变为bi-ai%bi
sum[id] %= b[id];//ai变为ai%bi
return ;
}
int left = lid(id), right = rid(id);
if(lg[id]==cl&&rg[id]==cr)//遇到匹配的区间
{
low[id]--; sum[id]++;
if(low[id]>)return ;
if ( low[id]<=)//如果low<=0,说明应该往下更新
{
//下面的更新一定是整个结点的,调用addup而不是add
addup(left, sum[id]);
addup(right, sum[id]);
//回溯时更新low和ans
low[id] = min(low[left], low[right]);
ans[id] = ans[left] + ans[right];
//因为sum已经加到子区间上了,把sum改为0
sum[id] =;
return ;
}
} //匹配区间结点
int mid = (lg[id] + rg[id]) >> ;
pushdown(id);
if (cr <= mid)
{
add(left, cl, cr, );
ans[id] = ans[left] + ans[right];
low[id] = min(low[left], low[right]);
return;
}
else if (cl > mid)
{
add(right, cl, cr, );
ans[id] = ans[left] + ans[right];
low[id] = min(low[left], low[right]);
return;
}
else
{
add(left, cl, mid,);
add(right, mid + , cr,);
ans[id] = ans[left] + ans[right];
low[id] = min(low[left], low[right]);
return;
}
} int qry(int id,int cl,int cr)
{
if (lg[id] == cl&&rg[id] == cr)
{
return ans[id];
} int mid = (lg[id] + rg[id]) >> ;
pushdown(id);
if (cr <= mid)
{
return qry(lid(id), cl, cr);
}
else if (cl > mid)
{
return qry(rid(id), cl, cr) ;
}
else
{
return qry(lid(id), cl, mid)+qry(rid(id), mid + , cr);
}
} int n,q;
char str[];
int main() { while (~scanf("%d %d" , &n,&q))
{
memset(lg, , sizeof lg);
memset(rg, , sizeof rg);
memset(low, , sizeof low);
memset(sum, , sizeof sum);
memset(ans, , sizeof ans);
build(, ,n);
int a1, a2;
for(int i=;i<q;++i)
{
scanf("%s", str);
scanf("%d %d", &a1, &a2);
if (str[] == 'a')
add(, a1, a2,);
else if (str[] == 'q')
printf("%d\n",qry(, a1, a2));
}
}
return ;
}
2018 杭电多校2 - Naive Operations的更多相关文章
- hdu6312 2018杭电多校第二场 1004 D Game 博弈
Game Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- 2018 杭电多校3 - M.Walking Plan
题目链接 Problem Description There are $$$n$$$ intersections in Bytetown, connected with $$$m$$$ one way ...
- 2018 杭电多校1 - Chiaki Sequence Revisited
题目链接 Problem Description Chiaki is interested in an infinite sequence $$$a_1,a_2,a_3,...,$$$ which i ...
- 2018 杭电多校1 - Distinct Values
题目链接 Problem Description Chiaki has an array of n positive integers. You are told some facts about t ...
- 2018杭电多校第二场1003(DFS,欧拉回路)
#include<bits/stdc++.h>using namespace std;int n,m;int x,y;int num,cnt;int degree[100007],vis[ ...
- 2018杭电多校第六场1009(DFS,思维)
#include<bits/stdc++.h>using namespace std;int a[100010];char s[20];int zhiren[100010];vector& ...
- 2018杭电多校第五场1002(暴力DFS【数位】,剪枝)
//never use translation#include<bits/stdc++.h>using namespace std;int k;char a[20];//储存每个数的数值i ...
- 2018杭电多校第三场1003(状态压缩DP)
#include<bits/stdc++.h>using namespace std;const int mod =1e9+7;int dp[1<<10];int cnt[1& ...
- 可持久化线段树的学习(区间第k大和查询历史版本的数据)(杭电多校赛第二场1011)
以前我们学习了线段树可以知道,线段树的每一个节点都储存的是一段区间,所以线段树可以做简单的区间查询,更改等简单的操作. 而后面再做有些题目,就可能会碰到一种回退的操作.这里的回退是指回到未做各种操作之 ...
随机推荐
- linux (rm指令) 及误删除解决
今天在群里看见这一幕: 看到这儿,我们学习一下 这个RM指令 rm命令可以删除一个目录中的一个或多个文件或目录,也可以将某个目录及其下属的所有文件及其子目录均删除掉.对于链接文件,只是删除整个链接文件 ...
- dubbo之监控中心(monitor)
一.monitor是dubbo框架中的一个监控中心.这个只是针对于消费者和提供者进行一个数据记录,不参与业务和使用.当然当monitor挂掉之后,也不会影响服务的正常运行. 二.在阿里的dubbo中也 ...
- 【原创】linux命令-Axel命令 - linux多线程下载 - 费元星 - 未来星开发团队
[费元星版权Q:9715234] Axel 是 Linux 下一个不错的HTTP/FTP高速下载工具.支持多线程下载.断点续[费元星版权Q:9715234]传,且可以从多个地址或者从一个地址的多个连接 ...
- 开发发布npm module包
开发发布npm module包 问题 在项目开发过程中,每当进入一个新的业务项目,从零开始搭建一套前端项目结构是一件让人头疼的事情,就要重新复制一个上一个项目的前端框架和组件代码库.其中很多功能的模块 ...
- RSA加密通信小结(一)
一.背景描述 帮朋友完成相关方案的改进. 二.计划与方案 1.加密方式采用RSA 1024加密. 2.发送与接收都采用RSA加密,采用两套不同的密钥. 3.统一的加解码函数.(此处除了对于传输数据进行 ...
- 关于Python 中的 if 语句
学习Python,最开始我们都是先从函数学起,Python教程中有很多函数,if算是其中之一. 可能最为人所熟知的编程语句就是 if 语句了.例如: >>> >>> ...
- 【WXS】简要介绍说明
WXS(WeiXin Script)是小程序的一套脚本语言. WXS有二种写法: 1) 以<wxs>标签书写脚本: 语法: <wxs module="[String]&qu ...
- idea clion编译器
RNMV64P0LA-eyJsaWNlbnNlSWQiOiJSTk1WNjRQMExBIiwibGljZW5zZWVOYW1lIjoiY24gdHUiLCJhc3NpZ25lZU5hbWUiOiIiL ...
- #pragma pack(n)对齐格式
#pragma pack(n)对齐格式 #pragma pack(n) 是预处理器用来指定对齐格式的指令,表示n对齐.当元素字节小于n时,要扩展到n:若元素字节大于n则占用其实际大小. struct ...
- iis 10 重新注册iis
iis 10 使用该命令 提示 版本不支持 C:\WINDOWS\system32>c:\windows\microsoft.net\framework64\v4.0.30319\aspnet_ ...