题目大意:初始给定 N 个点,支持三种操作:两点之间连边;一个点与一个连续区间编号的点之间连边;一个连续区间内的点和一个点连边,求执行 N 次操作之后的单源最短路。

题解:学会了线段树优化建图。

发现若暴力进行连边,时间和空间都会被卡到 \(O(n^2)\),直接起飞。

发现连边的点的编号是连续的,结合线段树可以维护连续区间信息的思想,就产生了线段树优化建图的方法。

在初始的 N 个节点的基础上建立两棵线段树,分别表示入树和出树,其中入树的上的各个节点允许单个节点的连接;出树上的节点允许连接单个节点。对于入树中的每个节点来说,需要连一条边权为 0 的有向边到它的两个儿子,表示若有节点连向父节点,那么一定也连向了儿子节点;对于出树上的每个节点来说,需要连一条边权为 0 的有向边到它的父节点,表示若当前节点可以连向某个节点,那么其子节点也一定可以走到这个节点。

时空复杂度分析:空间为两棵线段树的空间减去一份叶子节点的大小,即:\(3 * n\),时间复杂度为 \(elog^2v\),其中 e 是边的条数,v 是节点的个数。

代码如下

#include <bits/stdc++.h>
using namespace std;
typedef long long LL; struct edge {
int to;
LL w;
edge(int x = -1, int y = -1) {
to = x, w = y;
}
}; struct segtree {
#define ls(o) tree[o].lc
#define rs(o) tree[o].rc
struct node {
int lc, rc;
};
vector<node> tree;
int tot, rootin, rootout;
segtree(int n, vector<vector<edge>> &adj) {
tot = n;
tree.resize(3 * n);
buildin(rootin, 1, n, adj);
buildout(rootout, 1, n, adj);
}
void buildin(int &o, int l, int r, vector<vector<edge>> &adj) {
if (l == r) {
o = l;
return;
}
o = ++tot;
int mid = l + r >> 1;
buildin(ls(o), l, mid, adj);
buildin(rs(o), mid + 1, r, adj);
adj[o].emplace_back(ls(o), 0);
adj[o].emplace_back(rs(o), 0);
}
void buildout(int &o, int l, int r, vector<vector<edge>> &adj) {
if (l == r) {
o = l;
return;
}
o = ++tot;
int mid = l + r >> 1;
buildout(ls(o), l, mid, adj);
buildout(rs(o), mid + 1, r, adj);
adj[ls(o)].emplace_back(o, 0);
adj[rs(o)].emplace_back(o, 0);
}
void link(int o, int l, int r, int x, int y, int u, int w, int opt, vector<vector<edge>> &adj) {// opt 2 -> in 3 -> out
if (l == x && r == y) {
opt == 2 ? adj[u].emplace_back(o, w) : adj[o].emplace_back(u, w);
return;
}
int mid = l + r >> 1;
if (y <= mid) {
link(ls(o), l, mid, x, y, u, w, opt, adj);
} else if (x > mid) {
link(rs(o), mid + 1, r, x, y, u, w, opt, adj);
} else {
link(ls(o), l, mid, x, mid, u, w, opt, adj);
link(rs(o), mid + 1, r, mid + 1, y, u, w, opt, adj);
}
}
}; int main() {
ios::sync_with_stdio(0);
cin.tie(0), cout.tie(0);
int n, m, s;
cin >> n >> m >> s;
vector<vector<edge>> adj(3 * n);
segtree t(n, adj);
while (m--) {
int opt;
cin >> opt;
if (opt == 1) {
int u, v, w;
cin >> u >> v >> w;
adj[u].emplace_back(v, w);
} else {
int u, l, r, w;
cin >> u >> l >> r >> w;
t.link(opt == 2 ? t.rootin : t.rootout, 1, n, l, r, u, w, opt, adj);
}
}
vector<LL> d(3 * n, 1e18);
vector<bool> expand(3 * n);
priority_queue<pair<LL, int>> q;
auto dijkstra = [&]() {
d[s] = 0, q.push(make_pair(0, s));
while (!q.empty()) {
int u = q.top().second;
q.pop();
if (expand[u] == 1) {
continue;
}
expand[u] = 1;
for (auto e : adj[u]) {
int v = e.to, w = e.w;
if (d[v] > d[u] + w) {
d[v] = d[u] + w;
q.push(make_pair(-d[v], v));
}
}
}
};
dijkstra();
for (int i = 1; i <= n; i++) {
if (d[i] == 1e18) {
cout << "-1" << " ";
} else {
cout << d[i] << " ";
}
}
return 0;
}

【CF786B】Legacy的更多相关文章

  1. 【翻译】Sencha Touch2.4 The Layout System 布局

    [翻译]The Layout System 布局 In Sencha Touch there are two basic building blocks: componentsand containe ...

  2. 【转】在RedHat上搭建自己Email服务器

    原文:http://6839976.blog.51cto.com/6829976/1323482 by LN__@linux 目前邮件服务器中,想要拥有自己的邮件服务器,单单使用senmail,pos ...

  3. 在前端页面中使用@font-face来显示web自定义字体【转】

    本文转自W3CPLUS 的<CSS @font-face> @font-face是CSS3中的一个模块,他主要是把自己定义的Web字体嵌入到你的网页中,随着@font-face模块的出现, ...

  4. 【翻译】Flume 1.8.0 User Guide(用户指南) Processors

    翻译自官网flume1.8用户指南,原文地址:Flume 1.8.0 User Guide 篇幅限制,分为以下5篇: [翻译]Flume 1.8.0 User Guide(用户指南) [翻译]Flum ...

  5. 【翻译】Flume 1.8.0 User Guide(用户指南) source

    翻译自官网flume1.8用户指南,原文地址:Flume 1.8.0 User Guide 篇幅限制,分为以下5篇: [翻译]Flume 1.8.0 User Guide(用户指南) [翻译]Flum ...

  6. 【教程】在UEFI启动方式下,通过GRUB2引导,直接从硬盘ISO文件安装Windows10和Ubuntu双系统

    本文为作者原创,允许转载,但必须注明原文地址: https://www.cnblogs.com/byronxie/p/9949789.html 动机 最近在自学MIT6.828 Operating S ...

  7. 【转】PEP8 规范

    [转]PEP8 规范 Python PEP8 编码规范中文版   原文链接:http://legacy.python.org/dev/peps/pep-0008/ item detail PEP 8 ...

  8. 【转】CSS3属性 @font-face 整理

    原文: http://www.w3cplus.com/content/css3-font-face 出自: w3cplus.com 一.语法规则 @font-face { font-family: & ...

  9. 【Java】-NO.20.Exam.1.Java.1.001- 【1z0-807】- OCEA

    1.0.0 Summary Tittle:[Java]-NO.20.Exam.1.Java.1.001-[1z0-807] Style:EBook Series:Java Since:2017-10- ...

随机推荐

  1. Min swaps to sort array

    Given an array with distinct numbers, return an integer indicating the minimum number of swap operat ...

  2. ES-基本概念

    目录: 1.  ElasticSearch是什么 2. 基本概念 3. 几个关键词   正文: 1.  ElasticSearch是什么     ElasticSearch是一个基于Lucene的搜索 ...

  3. 阿里云ECS云服务器CentOS部署个人网站

    ping了一下coding pages和阿里云服务器的速度,意外感觉coding的速度反而更快.不过为了折腾,还是把博客迁移到阿里云,跌跌撞撞遇到很多坑,大多是由于对指令不熟悉以及部分教程省略了部分步 ...

  4. Nginx的一些常用配置

    #定义Nginx运行的用户和用户组 #user nobody; #nginx进程数,建议设置为等于CPU总核心数. worker_processes 1; #全局错误日志定义类型,[ debug | ...

  5. 解决Spring Boot集成Shiro,配置类使用Autowired无法注入Bean问题

    如题,最近使用spring boot集成shiro,在shiroFilter要使用数据库动态给URL赋权限的时候,发现 @Autowired 注入的bean都是null,无法注入mapper.搜了半天 ...

  6. Nmap 常用命令语法

    Nmap是一个网络连接端扫描软件,用来扫描网上电脑开放的网络连接端,确定哪些服务运行在哪些连接端,并且推断计算机运行哪个操作系统,正如大多数被用于网络安全的工具,Nmap也是不少黑客及骇客爱用的工具, ...

  7. golang 客户端

    package main import ( "fmt" "io/ioutil" "net/http" ) func main() { fmt ...

  8. Spring实战(十三)Spring事务

    1.什么是事务(Transaction)? 事务是指逻辑上的一组操作,要么全部成功,要么全部失败. 事务是指将一系列数据操作捆绑成为一个整体进行统一管理.如果某一事务执行成功,则该事务中进行的所有数据 ...

  9. Codeforces Round #406 (Div. 2) A MONSTER

    A. The Monster time limit per test 1 second memory limit per test 256 megabytes input standard input ...

  10. WPF 遍历资源字典中的控件

    object obItem=this.FindResource("canvasdt"); if (obItem is System.Windows.DataTemplate) { ...