[题解] CF786B Legacy
前言
题意
有 \(n\) 个点,\(q\) 次连边,以及起点 \(s\) 。连边具体分三种:
- \(1\) \(v\) \(u\) \(w\) 从 \(v\) 到 \(u\) 连一条边。
- \(2\) \(v\) \(l\) \(r\) \(w\) 从 \(v\) 向 \(l\) 到 \(r\) 所有点连一条边。
- \(3\) \(v\) \(l\) \(r\) \(w\) 从 \(l\) 到 \(r\) 所有点向 \(v\) 连一条边。
求所有点的最短路。
思路
以前做过一道比较类似的题:T'ill It's Over 都是线段树优化连边。
建立两棵线段树记为 \(A\) , \(B\) ,线段树 \(A\) 的所有结点向上连结父节点,线段树 \(B\) 的所有节点向下连结自己的子节点。
那么操作就可以看成 \(A\) 树对应区间连向 \(B\) 树对应区间,就转换为普通的线段树区间查询了。
现在需要考虑如何将 \(B\) 树的状态转换到 \(A\) 树上。
只需要将 \(B\) 树的结点连向 \(A\) 的对应结点就好了。
由于状态最后都会回归到 \(A\) 树中,所以最后查询 \(A\) 树中叶子结点区间为 \([i,i](i\in [1,n])\) 的点的最短路径,即为答案。
算法总时间复杂度为 \(O(n\log n*log((n+q)\log n))\) ,大概就是 \(O(n\log^2n)\) 级别的。
Code
#include <queue>
#include <cstdio>
#include <cstring>
using namespace std;
namespace Quick_Function {
template <typename Temp> void Read(Temp &x) {
x = 0; char ch = getchar(); bool op = 0;
while(ch < '0' || ch > '9') { if(ch == '-') op = 1; ch = getchar(); }
while(ch >= '0' && ch <= '9') { x = (x << 1) + (x << 3) + (ch ^ 48); ch = getchar(); }
if(op) x = -x;
}
template <typename T, typename... Args> void Read(T &t, Args &... args) { Read(t); Read(args...); }
template <typename Temp> Temp Max(Temp x, Temp y) { return x > y ? x : y; }
template <typename Temp> Temp Min(Temp x, Temp y) { return x < y ? x : y; }
template <typename Temp> Temp Abs(Temp x) { return x < 0 ? (-x) : x; }
template <typename Temp> void Swap(Temp &x, Temp &y) { x ^= y ^= x ^= y; }
}
using namespace Quick_Function;
#define INF 1e17
#define int long long
const int MAXN = 5e5 + 5;
const int MAXM = 5e6 + 5;
struct Edge { int To, Dist, Next; };
int head[MAXN], edgetot = 1;
Edge edge[MAXM];
void Addedge(int u, int v, int w) { edge[++edgetot].Next = head[u], edge[edgetot].To = v, edge[edgetot].Dist = w, head[u] = edgetot; }
struct Node {
int id, dis;
Node() {}
Node(int I, int D) { id = I, dis = D; }
friend bool operator < (Node x, Node y) { return x.dis > y.dis; }
};
priority_queue<Node> que;
int dis[MAXN];
bool vis[MAXN];
int n, q, s, p;
struct Segment_Node {
int l, r, id;
#define LS(x) (x << 1)
#define RS(x) (x << 1 | 1)
};
struct Segment_Tree {
Segment_Node t[MAXN << 2][2];//0为A线段树,1为B线段树
int tot;
void Build(int pos, int l, int r, int flag) {//初始化建树
t[pos][flag].l = l, t[pos][flag].r = r, t[pos][flag].id = ++tot;
if(l == r) {
if(l == s && !flag) p = t[pos][flag].id;
if(flag) Addedge(t[pos][1].id, t[pos][0].id, 0);//这里忘写调了半个小时QAQ
return;
}
int mid = (l + r) >> 1;
Build(LS(pos), l, mid, flag); Build(RS(pos), mid + 1, r, flag);
if(!flag) {
Addedge(t[LS(pos)][flag].id, t[pos][flag].id, 0);
Addedge(t[RS(pos)][flag].id, t[pos][flag].id, 0);
}
else {
Addedge(t[pos][flag].id, t[LS(pos)][flag].id, 0);
Addedge(t[pos][flag].id, t[RS(pos)][flag].id, 0);
Addedge(t[pos][1].id, t[pos][0].id, 0);
}
}
int Query(int pos, int x, int flag) {//查询区间为[x,x]的叶子结点
if(t[pos][flag].l == t[pos][flag].r) return t[pos][flag].id;
if(x <= t[LS(pos)][flag].r) return Query(LS(pos), x, flag);
else return Query(RS(pos), x, flag);
}
void Update1(int pos, int l, int r, int x, int w) {//一连多
if(l <= t[pos][1].l && t[pos][1].r <= r) {
Addedge(Query(1, x, 0), t[pos][1].id, w);
return;
}
if(l <= t[LS(pos)][1].r) Update1(LS(pos), l, r, x, w);
if(r >= t[RS(pos)][1].l) Update1(RS(pos), l, r, x, w);
}
void Update2(int pos, int l, int r, int x, int w) {//多连一
if(l <= t[pos][0].l && t[pos][0].r <= r) {
Addedge(t[pos][0].id, Query(1, x, 1), w);
return;
}
if(l <= t[LS(pos)][0].r) Update2(LS(pos), l, r, x, w);
if(r >= t[RS(pos)][0].l) Update2(RS(pos), l, r, x, w);
}
};
Segment_Tree tree;
void Dijkstra() {//最短路
for(int i = 1; i <= tree.tot; i++) dis[i] = INF;
que.push(Node(p, 0)); dis[p] = 0;
while(!que.empty()) {
int u = que.top().id; que.pop();
if(vis[u]) continue;
vis[u] = 1;
for(int i = head[u]; i; i = edge[i].Next) {
int v = edge[i].To;
if(dis[v] > dis[u] + edge[i].Dist) {
dis[v] = dis[u] + edge[i].Dist;
que.push(Node(v, dis[v]));
}
}
}
}
signed main() {
Read(n, q, s);
tree.Build(1, 1, n, 0); tree.Build(1, 1, n, 1);
for(int i = 1, opt, u, v, l, r, w; i <= q; i++) {
Read(opt, u);
if(opt == 1) {
Read(v, w);
tree.Update1(1, v, v, u, w);
}
else if(opt == 2) {
Read(l, r, w);
tree.Update1(1, l, r, u, w);
}
else {
Read(l, r, w);
tree.Update2(1, l, r, u, w);
}
}
Dijkstra();
for(int i = 1; i <= n; i++) printf("%lld ", dis[tree.Query(1, i, 0)] == INF? -1 : dis[tree.Query(1, i, 0)]);
return 0;
}
[题解] CF786B Legacy的更多相关文章
- CF786B Legacy(线段树优化建边)
模板题CF786B Legacy 先说算法 如果需要有n个点需要建图 给m个需要建边的信息,从单点(或区间内所有点)向一区间所有点连边 如果暴力建图复杂度\(mn^2\) 以单点连向区间为例,在n个点 ...
- CF786B Legacy && 线段树优化连边
线段树优化连边 要求点 \(x\) 向区间 \([L, R]\) 连边, 一次的复杂度上限为 \(O(n)\) 然后弄成线段树的结构 先父子连边边权为 \(0\) 这样连边就只需要连父亲就可以等效于连 ...
- 题解 CF786B 【Legacy】
本题要求我们支持三种操作: ① 点向点连边. ② 点向区间连边. ③ 区间向点连边. 然后跑最短路得出答案. 考虑使用线段树优化建图. 建两颗线段树,入树和出树,每个节点为一段区间的原节点集合.入树内 ...
- CF786B Legacy 线段树优化建图
问题描述 CF786B LG-CF786B 题解 线段树优化建图 线段树的一个区间结点代表 \([l,r]\) 区间点. 然后建立区间点的时候就在线段树上建边,有效减少点的个数,从而提高时空效率. 优 ...
- 线段树优化建图 || CF786B Legacy
题面:786B - Legacy 代码: #include<cstdio> #include<cstring> #include<iostream> #includ ...
- CF786B Legacy 线段树优化建图 + spfa
CodeForces 786B Rick和他的同事们做出了一种新的带放射性的婴儿食品(???根据图片和原文的确如此...),与此同时很多坏人正追赶着他们.因此Rick想在坏人们捉到他之前把他的遗产留给 ...
- CF786B Legacy(线段树优化建图)
嘟嘟嘟 省选Day1T2不仅考了字符串,还考了线段树优化建图.当时不会,现在赶快学一下. 线段树能优化的图就是像这道题一样,一个点像一个区间的点连边,或一个区间像一个点连边.一个个连就是\(O(n ^ ...
- CF786B Legacy
思路 线段树优化建图 基本思想就是要把一个区间连边拆成log个节点连边, 然后一颗入线段树,一颗出线段树,出线段树都由子节点向父节点连边(可以从子区间出发),入线段树从父节点向子节点连边(可以到达子区 ...
- 【CF786B】Legacy
题目大意:初始给定 N 个点,支持三种操作:两点之间连边:一个点与一个连续区间编号的点之间连边:一个连续区间内的点和一个点连边,求执行 N 次操作之后的单源最短路. 题解:学会了线段树优化建图. 发现 ...
随机推荐
- Azure Cost alerts 费用成本分析
一,引言 依稀记得在一月初,我们在 Azure 上做成了一个 费用警报的监控,果不其然,前两天 Azure 给我发了两封封 Azure 预警警报的邮件,提醒我的预算的总费用超过了设置的通知阈值 &qu ...
- CRC校验原理和verilog实现方法(一)
1.CRC简介 CRC全称循环冗余校验(Cyclic Redundancy Check, CRC),是通信领域数据传输技术中常用的检错方法,用于保证数据传输的可靠性.网上有关这方面的博客和资料很多,本 ...
- Apache JMeter 5.4.1 Build Development
1. 说明 经过漫长的等待终于将开发环境搭建成功了!网络慢真的是伤不起!grade,确实要比maven简洁.....嗯!真香! 2. 工具准备 JDK1.8+ 这... ...
- 使用C# (.NET Core) 实现装饰模式 (Decorator Pattern) 并介绍 .NET/Core的Stream
该文章综合了几本书的内容. 某咖啡店项目的解决方案 某咖啡店供应咖啡, 客户买咖啡的时候可以添加若干调味料, 最后要求算出总价钱. Beverage是所有咖啡饮料的抽象类, 里面的cost方法是抽象的 ...
- 最小生成树(Prim算法,Kruskal算法 )
声明:图片及内容基于https://www.bilibili.com/video/BV1yp4y1Q74o?from=articleDetail 最小生成树原理 . 普利姆(Prim)算法 原理 Pr ...
- [Fundamental of Power Electronics]-PART I-2.稳态变换器原理分析-2.3 Boost 变换器实例
2.3 Boost 变换器实例 图2.13(a)所示的Boost变换器器是另一个众所周知的开关模式变换器,其能够产生幅值大于直流输入电压的直流输出电压.图2.13(b)给出了使用MOSFET和二极管的 ...
- Java字节流和字符流,是时候总结一下IO流了
目录 从接收输入值说起 字节流读取 字符流读取 Scanner 读取 什么是 IO 流 字节流和字符流 字节流 字节输入流 字节输出流 缓冲流的原理 字符流 字符输入流 字符输出流 为什么字符流需要 ...
- java.net.BindException: Problem binding to [hadoop103:8031] java.net.BindException
ResourceManger启动失败,Namenode启动成功,这个问题排查了好久 在hadoop-2.7.6/logs/yarn-root-resourcemanager-hadoop102.log ...
- 【ProLog - 4.0 List】
[简介] 列表是Prolog编程中常用的一种重要的递归数据结构 列表是一个有限的元素序列 实例: 所有Prolog术语都可以是列表的元素,一个非空的List应该含有两个元素:头元素(Head)和尾元素 ...
- sql 如何删除(代替)字段内某一部分内容
方法一(此方法既可用于删除某一列字段中的某部分字符,也可用于替换某一列字段中的某部分字符) update Table_Name set Column_Name=replace(Column_Name, ...