problem

\(\mathtt {loj-3146}\)

题意概要:一条直线上有 \(n+1\) 个点和 \(n\) 条道路,每条道路连通相邻两个点。在 \(q\) 个时刻内,每个时刻有如下两种操作之一:

  • 切换某条道路的状态,即:若原来是连通的,则现在断开;若原来断开,则现在连通
  • 给出 \(x,y\),询问在这次询问之前,有多少个时刻满足 \(a\rightarrow b\) 的道路连通(即这一段的道路都连通)

\(n,q\leq 3\times 10^5\),时限 \(5s\)

Solution

切了前两题让我还以为今年apio能ak的说,这题没切主要是因为没有想到可以将若干段区间的和变为所有右端点的坐标减去所有左端点的坐标,之前一直在想如何计算修改对答案的贡献来着

在这题里,连接即一段存在区间的左端点,减去当前时刻 \(t\),断开即一段区间的右端点,加上当前时刻 \(t\)。特别的,当一次询问时若他们之间连通,则需要再次强行加上当前时刻 \(t\)。

考虑切换道路 \((x,x+1)\),找到 \(x\) 往左走的最远端 \(l\),与 \(x+1\) 往右走的最远端 \(r\),则这次切换的影响为:左端点在 \([l,x]\) 内,且右端点在 \([x+1,r]\) 内的所有区间。

放到平面上去就是一个矩形,而询问就是询问这个平面上的一个点。有时间、\(x\)、\(y\)共三维,用 \(CDQ+BIT\) 可做到 \(O(n\log^2n)\)。

Code

至于找到每个位置向左向右的最远点,用 \(set\) 维护一下所有的极长道路区间即可

//loj-3146
#include <bits/stdc++.h>
using namespace std;
#define lb(x) (x&(-x)) template <typename _tp> inline void read(_tp&x) {
char ch=getchar();x=0;while(!isdigit(ch))ch=getchar();
while(isdigit(ch))x=x*10+ch-'0',ch=getchar();
} const int N = 301000;
int n; namespace WK {
namespace BIT {
int d[N];
inline void add(int x, int v) {for(;x<=n;x+=lb(x)) d[x] += v;}
inline int qry(int x) {int r=0;for(;x;x^=lb(x))r+=d[x];return r;}
} struct node {
int op, x, y, v;
} q[N*4], b[N*4];
int Ans[N], Qs, tot; void qwq(int l, int r) {
if(l == r) return ;
int m = l + r >> 1;
qwq(l, m), qwq(m+1, r);
int t0 = l, t1 = m + 1;
int tt = l;
while(t0 <= m or t1 <= r) {
if((t0 <= m and t1 <= r and q[t0].x <= q[t1].x) or t1 > r) {
if(q[t0].op == 0)
BIT::add(q[t0].y, q[t0].v);
b[tt++] = q[t0++];
} else {
if(q[t1].op == 1)
Ans[q[t1].v] += BIT::qry(q[t1].y);
b[tt++] = q[t1++];
}
}
for(int i=l;i<=m;++i)
if(q[i].op == 0)
BIT::add(q[i].y, -q[i].v);
for(int i=l;i<=r;++i)
q[i] = b[i];
} void work() {
qwq(1, tot);
for(int i=1;i<=Qs;++i)
printf("%d\n", Ans[i]);
} inline void add_modify(int x0, int x1, int y0, int y1, int v) {
q[++tot] = (node) {0, x0, y0, +v};
q[++tot] = (node) {0, x0, y1+1, -v};
q[++tot] = (node) {0, x1+1, y0, -v};
q[++tot] = (node) {0, x1+1, y1+1, +v};
}
inline void add_query(int x, int y, int vl) {
q[++tot] = (node) {1, x, y, ++Qs};
Ans[Qs] = vl;
}
} char str[N]; bool st[N];
int Q; typedef pair<int,int> pii;
set <pii> c;
set <pii> :: iterator it, itr;
#define ins insert
#define ers erase namespace BIT {
int d[N];
inline void add(int x, int v) {for(;x<=n;x+=lb(x)) d[x] += v;}
inline int qry(int l, int r) {
int res = 0;
for(;l;l^=lb(l)) res -= d[l];
for(;r;r^=lb(r)) res += d[r];
return res;
}
} int main() {
read(n), read(Q);
scanf("%s", str+1);
str[++n] = '0';
for(int i=1;i<=n;++i) {
st[i] = str[i] == '1';
if(st[i]) BIT::add(i+1, +1);
}
for(int i=1,j;(j=i)<=n;i=j+1) {
while(j < n and st[j]) ++j;
c.ins(pii(i, j));
} char opt[7];
int x, y;
for(int i=1;i<=Q;++i) {
scanf("%s", opt);
if(opt[0] == 't') {
read(x);
if(st[x]) {
itr = c.upper_bound(pii(x, n+1)), --itr;
int l = itr->first, r = itr->second;
WK::add_modify(l, x, x+1, r, +i);
c.ers(itr), c.ins(pii(l, x)), c.ins(pii(x+1, r));
BIT::add(x+1, -1);
st[x] = false;
} else {
it = itr = c.upper_bound(pii(x, n+1)), --it;
int l = it->first, r = itr->second;
WK::add_modify(l, x, x+1, r, -i);
c.ers(it), c.ers(itr), c.ins(pii(l, r));
BIT::add(x+1, +1);
st[x] = true;
}
} else {
read(x), read(y);
WK::add_query(x, y, BIT::qry(x, y) == y-x ? +i : 0);
}
}
WK::work();
return 0;
}

题解-APIO2019路灯的更多相关文章

  1. 【LOJ#3146】[APIO2019]路灯(树套树)

    [LOJ#3146][APIO2019]路灯(树套树) 题面 LOJ 题解 考场上因为\(\text{bridge}\)某个\(\text{subtask}\)没有判\(n=1\)的情况导致我卡了\( ...

  2. P5445 [APIO2019]路灯(树套树)

    P5445 [APIO2019]路灯 转化为平面上的坐标(x,y),set维护连续区间. 用树套树维护矩阵加法,单点查询. 注意维护矩阵差分的时候, $(x,y,v)$是对$(x,y)(n+1,n+1 ...

  3. LOJ3146 APIO2019路灯(cdq分治+树状数组)

    每个时刻都形成若干段满足段内任意两点可达.将其视为若干正方形.则查询相当于求历史上某点被正方形包含的时刻数量.并且注意到每个时刻只有O(1)个正方形出现或消失,那么求出每个矩形的出现时间和消失时间,就 ...

  4. 题解-APIO2019桥梁

    problem \(\mathrm {loj-3145}\) 题意概要:给定一张 \(n\) 点 \(m\) 边的无向图,边有边权,共 \(q\) 次操作,每次会将第 \(x\) 条边的权值改为 \( ...

  5. 题解-APIO2019奇怪装置

    problem loj-3144 题意概要:设函数 \(f(t)\) 的返回值为一个二元组,即 \(f(t)=((t+\lfloor \frac tB\rfloor)\bmod A, t\bmod B ...

  6. P5445 [APIO2019]路灯

    传送门· 对于询问 $(a,b)$ ,感觉一维很不好维护,考虑把询问看成平面上的一个点,坐标为 $(a,b)$ 每个坐标 $(x,y)$ 的值表示到当前 $x$ 和 $y$ 联通的时间和 考虑一个修改 ...

  7. APIO2019简要题解

    Luogu P5444 [APIO2019]奇怪装置 看到这种题,我们肯定会想到\((x,y)\)一定有循环 我们要找到循环节的长度 推一下发现\(x\)的循环节长为\(\frac{AB}{B+1}\ ...

  8. 【APIO2019】路灯(ODT & (树套树 | CDQ分治))

    Description 一条 \(n\) 条边,\(n+1\) 个点的链,边有黑有白.若结点 \(a\) 可以到达 \(b\),需要满足 \(a\to b\) 的路径上的边不能有黑的.现给出 \(0\ ...

  9. 2021record

    2021-10-14 P2577 [ZJOI2004]午餐 2021-10-13 CF815C Karen and Supermarket(小小紫题,可笑可笑) P6748 『MdOI R3』Fall ...

随机推荐

  1. 制作 python解释器

    https://www.zhihu.com/tardis/sogou/qus/27286136

  2. JQuery事件绑定bind、live、on、trigger

    one 作用:只触发一次,并在触发后失效,触发时会产生时间冒泡. 语法:$(selector).one(event,data,function) 例子: $(item).one("click ...

  3. 001 centos7中的安装

    以前学习过centos6.4的安装使用,然后新使用centos7,发现有些修改配置变化了,记录一下. 1.修改主机名 查看主机名: 修改: 二:ip的固定 1.参考文章 https://www.jia ...

  4. Flutter 中的常见的按钮组件 以及自定义按钮组件

    Flutter 里有很多的 Button 组件很多,常见的按钮组件有:RaisedButton.FlatButton. IconButton.OutlineButton.ButtonBar.Float ...

  5. Python 相对导入attempted relative import beyond top-level package

    ValueError: attempted relative import beyond top-level package 假设有如下层次包目录 project/ __init__.py mypac ...

  6. RabbitMQ 入门教程(PHP版) 简单Demo

    RabbitMQ的关键字说明 (1)Broker:经纪人.提供一种传输服务,维护一条从生产者到消费者的传输线路,保证消息数据能按照指定的方式传输.粗略的可以将图中的RabbitMQ Server当作B ...

  7. 【444】Data Analysis (shp, arcpy)

      ABS suburbs data of AUS 1. Dissolve Merge polygons with the same attribute of "SA2_NAME16&quo ...

  8. 123457123456#0#-----com.cym.shuXue02--前拼后广--开心学数学

    com.cym.shuXue02--前拼后广--开心学数学

  9. Jrebel激活方法(转)

    本次服务长期稳定提供给各位同学使用哦!服务器地址:https://jrebel.qekang.com/{GUID}在线GUID地址:在线生成GUID如果失效刷新GUID替换就可以!打开jrebel 激 ...

  10. docker容器中解决出现:^H^H^H^H

    docker容器中解决出现:^H^H^H^H 环境:docker容器是debain系统 解决: 把stty erase ^H 添加到.bash_profile中 vim /etc/profile st ...