\(\mathcal{Description}\)

  Link.

  给定序列 \(\{w_n\}\),选择 \(i\) 位置的代价为 \(w_i\),要求每个位置要不被选择,要不左右两个位置至少被选择一个。求前 \(k\) 小的选择代价。

  \(n,k\le2.5\times10^5\)。

\(\mathcal{Solution}\)

  建图,边形如 \(\lang i,i+j,w_i\rang~(j=1,2,3)\),再引入左右两个虚点 \(s,t\),那么每种方案对应从 \(s\) 到 \(t\) 的一条路径,题目即求无负权图上的前 \(k\) 小路径,是道模板题,很可惜我并不会。(

  还是用前 \(k\) 小的一贯套路:用堆维护,每次取当前最优方案扩展。首先在建出以 \(t\) 为根的内向最短路树,考虑用一条非树边 \(\lang u,v,w\rang\) 替换最短路 \(P=\lang s,\cdots,t\rang\) 的一段,则有 \(P'=\lang s,\cdots,u,v,\cdots,t\rang\),且满足 \(w(P')=d_S-d_u+w+d_v\),其中 \(d_u\) 即 \(u\) 到 \(t\) 的最短路。这样我们得到了扩展一次的方法:取 \(s\) 到 \(t\) 最短路的邻接边 \(\lang u,v,w\rang\) 中,\(d_v+w-d_u\) 最小的边替换最短路。

  如果替换多条边呢?可以发现有两种替换方法:

  • 将当前最优的 \(\lang u,v,w\rang\) 替换为新的 \(\lang u,v',w'\rang\);
  • 在当前 \(\lang u,v,w\rang\) 的基础上,取 \(v\) 到 \(t\) 最短路的邻接边扩展。

  那么用可持久化左偏树维护每个点到 \(t\) 的可用邻接边集合即可支持扩展。设图的点数为 \(n\),边数为 \(m\),整个复杂度则为 \(\mathcal O((m+k)\log m)\),本题则有 \(\mathcal O((n+k)\log n)\)。

\(\mathcal{Code}\)

/*~Rainybunny~*/

#include <bits/stdc++.h>

#define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i ) typedef long long LL;
typedef std::pair<LL, int> PLI;
typedef std::pair<int, int> PII;
#define fi first
#define se second const int MAXN = 2.5e5;
int n, m, pre[MAXN + 5], root[MAXN + 5];
LL dis[MAXN + 5];
std::vector<PII> oriG[MAXN + 5], revG[MAXN + 5]; struct PersistentLeftistTree {
static const int MAXND = 3e7;
int node, ch[MAXND + 5][2], rds[MAXND + 5];
PLI val[MAXND + 5]; // (cost, end point). inline int newnd( const PLI& v ) {
val[++node] = v, ch[node][0] = ch[node][1] = 0, rds[node] = 1;
return node;
} inline int merge( int x, int y ) {
if ( !x || !y ) return x | y;
if ( val[x] > val[y] ) std::swap( x, y );
int u = newnd( val[x] );
ch[u][0] = ch[x][0], ch[u][1] = merge( ch[x][1], y );
if ( rds[ch[u][1]] > rds[ch[u][0]] ) std::swap( ch[u][0], ch[u][1] );
return rds[u] = rds[ch[u][1]] + 1, u;
}
} plt; inline void link( const int s, const int t, const int w ) {
oriG[s].push_back( { t, w } ), revG[t].push_back( { s, w } );
// std::cerr << s << ' ' << t << ' ' << w << '\n';
} inline void dijkstra() {
static std::priority_queue<PLI, std::vector<PLI>, std::greater<PLI> > heap;
memset( dis, 0x3f, sizeof dis );
heap.push( { dis[n + 1] = 0, n + 1 } );
while ( !heap.empty() ) {
PLI p( heap.top() ); heap.pop();
if ( dis[p.se] < p.fi ) continue;
for ( const PII& e: revG[p.se] ) {
if ( dis[e.fi] > p.fi + e.se ) {
heap.push( { dis[e.fi] = p.fi + e.se, e.fi } );
pre[e.fi] = p.se;
}
}
}
} inline void solve() {
per ( u, n, 0 ) {
root[u] = root[pre[u]];
for ( const PII& v: oriG[u] ) {
if ( pre[u] != v.fi ) {
root[u] = plt.merge( root[u],
plt.newnd( { dis[v.fi] + v.se - dis[u], v.fi } ) );
}
}
} static std::priority_queue<PLI, std::vector<PLI>, std::greater<PLI> > heap;
std::cout << dis[0] << '\n';
if ( root[0] ) heap.push( { plt.val[root[0]].fi, root[0] } );
while ( --m ) {
if ( heap.empty() ) { std::cout << "-1\n"; continue; }
PLI p( heap.top() ); heap.pop();
std::cout << dis[0] + p.fi << '\n'; int u;
if ( ( u = plt.merge( plt.ch[p.se][0], plt.ch[p.se][1] ) ) ) {
heap.push( { p.fi - plt.val[p.se].fi + plt.val[u].fi, u } );
}
if ( ( u = root[plt.val[p.se].se] ) ) {
heap.push( { p.fi + plt.val[u].fi, u } );
}
}
} int main() {
std::ios::sync_with_stdio( false ), std::cin.tie( 0 ); std::cin >> n >> m;
rep ( i, 1, n ) {
int w; std::cin >> w;
rep ( j, 1, 3 ) {
if ( i + j <= n || ( i + j == n + 1 && j < 3 ) ) {
link( i, i + j, w );
}
}
}
link( 0, 1, 0 );
if ( n > 1 ) link( 0, 2, 0 ); dijkstra();
// rep ( i, 0, n + 1 ) std::cout << dis[i] << ' ' << pre[i] << '\n';
solve();
return 0;
}

Solution -「Gym 102979L」 Lights On The Road的更多相关文章

  1. Solution -「Gym 102979E」Expected Distance

    \(\mathcal{Description}\)   Link.   用给定的 \(\{a_{n-1}\},\{c_n\}\) 生成一棵含有 \(n\) 个点的树,其中 \(u\) 连向 \([1, ...

  2. Solution -「Gym 102956F」Find the XOR

    \(\mathcal{Description}\)   Link.   给定 \(n\) 个点 \(m\) 条边的连通无向图 \(G\),边有边权.其中 \(u,v\) 的距离 \(d(u,v)\) ...

  3. Solution -「Gym 102956B」Beautiful Sequence Unraveling

    \(\mathcal{Description}\)   Link.   求长度为 \(n\),值域为 \([1,m]\) 的整数序列 \(\lang a_n\rang\) 的个数,满足 \(\not\ ...

  4. Solution -「Gym 102956F」Border Similarity Undertaking

    \(\mathcal{Description}\)   Link.   给定一张 \(n\times m\) 的表格,每个格子上写有一个小写字母.求其中长宽至少为 \(2\),且边界格子上字母相同的矩 ...

  5. Solution -「Gym 102956A」Belarusian State University

    \(\mathcal{Description}\)   Link.   给定两个不超过 \(2^n-1\) 次的多项式 \(A,B\),对于第 \(i\in[0,n)\) 个二进制位,定义任意一个二元 ...

  6. Solution -「Gym 102798I」Sean the Cuber

    \(\mathcal{Description}\)   Link.   给定两个可还原的二阶魔方,求从其中一个状态拧到另一个状态的最小步数.   数据组数 \(T\le2.5\times10^5\). ...

  7. Solution -「Gym 102798K」Tree Tweaking

    \(\mathcal{Description}\)   Link.   给定排列 \(\{p_n\}\),求任意重排 \(p_{l..r}\) 的元素后,将 \(\{p_n\}\) 依次插入二叉搜索树 ...

  8. Solution -「Gym 102798E」So Many Possibilities...

    \(\mathcal{Description}\)   Link.   给定非负整数序列 \(\{a_n\}\) 和 \(m\),每次随机在 \(\{a\}\) 中取一个非零的 \(a_i\)(保证存 ...

  9. Solution -「Gym 102759I」Query On A Tree 17

    \(\mathcal{Description}\)   Link.   给定一棵含 \(n\) 个结点的树,结点 \(1\) 为根,点 \(u\) 初始有点权 \(a_u=0\),维护 \(q\) 次 ...

随机推荐

  1. vim - 显示不可见字符(:set list)

    默认情况下,vim是不会显示space,tabs,newlines,trailing space,wrapped lines等不可见字符的.我们可以使用以下命令打开list选项,来显示非可见字符: : ...

  2. Linux环境下的Docker的安装和部署、学习二

    DockerFile体系结构(保留字指令) FROM:基础镜像,当前新镜像是基于哪个镜像的 MAINTAINER:镜像维护者的姓名和邮箱地址 RUN:容器构建时需要运行的命令 EXPOSE:当前容器对 ...

  3. 【vps】如何在vps上安装mirai机器人?

    [vps]如何在vps上安装mirai机器人? 前言 由于某位师傅在群里设置了一个bot,吸引了我,所以我之前找他问了点bot的相关知识,这几天正好服务器搬迁,所以就在新服务器上再装一遍bot 1.安 ...

  4. 【记录一个问题】opencv中 cv::dft()与cv::ocl_dft()计算的结果相差较大

    以一个跟踪算法来测试: 使用cv::dft(), 矩阵未按照2次幂对齐,最终跟踪平均准确率 84.3% 使用cv::dft(),矩阵使用cv::copyMakeBorder对齐,最终跟踪平均准确率 8 ...

  5. Docker环境安装,基本命令集合

    一.docker安装 1).卸载旧的安装包 centos7默认安装的docker版本是1.13.1,卸载它,安装新的版本. root用户下,一次把这坨命令复制进去 yum remove docker ...

  6. 学习axios必知必会(1)~axios基本介绍、axios配置、json-server接口模拟工具

    一.axios基本介绍 1.axios(前端最流行的 ajax 请求库) 特点: ① 基于 xhr + promise 的异步 ajax 请求库 ② 浏览器端/node 端都可以使用 ③ 支持请求/响 ...

  7. http头文件

    http 文件头详解 HTTP(HyperTextTransferProtocol)是超文本传输协议的缩写, 它用于传送WWW方式的数据,关于HTTP协议的详细内容请参考RFC2616.HTTP协议采 ...

  8. 传统的DOM渲染方式?

    1.什么是DOM渲染? 所谓的DOM渲染是指的是对于浏览器中展现给用户的DOM文档的生成的过程. 2.DOM渲染的过程,大致可以分为三个阶段: --纯后端渲染 --纯前端渲染 --服务端的JS渲染结合 ...

  9. Nginx全面介绍 什么是Nginx?

    目录 一:Nginx全面讲解 1.简介: 2.nginx的用武之地 3.关于代理(解析含义作用) 二:正向代理 三:反向代理 四:项目应用场景 五:正向代理与反向代理区别 1.正向代理 2.反向代理 ...

  10. 关于使用学生或者教师身份免费使用JetBrains全家桶的说明

    官网操作 JetBrains是一家捷克的软件开发公司,该公司位于捷克的布拉格,并在俄罗斯的圣彼得堡及美国麻州波士顿都设有办公室,该公司有众多的好用的IDE,比如pycharm,webstorm,Int ...