题面

一道特殊的最短路题。

给一个

n

n

n 个点

m

m

m 条有向边的图,每条边上有数字

0

\tt0

0 或

1

\tt1

1 ,定义一个路径的长度为这个路径上依次经过的边上的数字拼在一起后在二进制下的值(前导

0

\tt0

0 对该路径长度没有贡献)。现在需要你求出从

1

1

1 号点出发,到

2

n

2\sim n

2∼n 号点的路径的长度的最小值。

n

1

0

6

,

m

2

1

0

6

.

n\leq10^6,m\leq2\cdot10^6.

n≤106,m≤2⋅106.

题解 & CODE

我们考虑该二进制路径的特性,前导

0

\tt0

0 是没有贡献的,因此

1

1

1 号点能直接通过

0

\tt0

0 走到的点肯定先处理出来。然后,路径上就必定有一个

1

\tt1

1 了,第一个

1

\tt1

1 的位数越高,路径长度就一定越大,所以接下来可以基于路径长度来简化该最短路。

我们根据首个

1

1

1 的位数来对图分层,此时进行 bfs,每次把该层的所有点拿出来,拿出来的时候就按照路径从小到大站成一列了,然后从前到后扩张每个点,优先走

0

0

0 边。这样可以线性地得到下一层所有点的路径长度,以及顺便把下一层的点按照路径长度从小到大放进了队列里。

时间复杂度

O

(

n

+

m

)

O(n+m)

O(n+m) 。


笔者的第二种方法仅供娱乐。

考虑取模过后(哈希过后)失去相对大小信息,无法比较,高精度又太慢。这个时候就可以用离散化

我们对

D

i

j

k

s

t

r

a

\rm Dijkstra

Dijkstra 进行改进,在求最短路的同时,求出每个点路径长度的排名(长度相等排名相同)。然后我们就可以依次比较 转移前驱的排名转移边 来对未确定的点进行择优,以及确定新排名时判断重复。

代码简单,只需要将传统 手写堆 优化

D

i

j

k

s

t

r

a

\rm Dijkstra

Dijkstra 进行小小的翻修。

时间复杂度

O

(

n

log

n

)

O(n\log n)

O(nlogn) 。输麻了

#include<set>
#include<map>
#include<queue>
#include<stack>
#include<random>
#include<vector>
#include<bitset>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 2000005
#define LL long long
#define DB double
#define lowbit(x) (-(x) & (x))
#define ENDL putchar('\n')
#define FI first
#define SE second
int xchar() {
static const int maxn = 100000;
static char b[maxn];
static int len = 0,pos = 0;
if(pos == len) pos = 0,len = fread(b,1,maxn,stdin);
if(pos == len) return -1;
return b[pos ++];
}
//#define getchar() xchar()
LL read() {
LL f=1,x=0;int s = getchar();
while(s < '0' || s > '9') {if(s<0) return -1;if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s^48);s = getchar();}
return f*x;
}
void putpos(LL x) {if(!x)return ;putpos(x/10);putchar('0'+(x%10));}
void putnum(LL x) {
if(!x) {putchar('0');return ;}
if(x<0) {putchar('-');x = -x;}
return putpos(x);
}
void AIput(LL x,int c) {putnum(x);putchar(c);} const int MOD = 1000000007;
int n,m,s,o,k;
int hd[MAXN],v[MAXN<<1],nx[MAXN<<1],w[MAXN<<1],cne;
void ins(int x,int y,int z) {
nx[++ cne] = hd[x]; v[cne] = y; w[cne] = z; hd[x] = cne;
}
int dp[MAXN],rk[MAXN];
int pr[MAXN],nw[MAXN];
int mg(int a,int b) {
if(pr[a] == pr[b]) return nw[a] < nw[b] ? a:b;
return pr[a] < pr[b] ? a:b;
}
int tre[MAXN<<1];
void upd(int x,int y) {
tre[n+x] = y;
for(int s = (n+x)>>1;s>0;s >>= 1) {
tre[s] = mg(tre[s<<1],tre[s<<1|1]);
}return ;
}
int main() {
freopen("path.in","r",stdin);
freopen("path.out","w",stdout);
n = read();m = read();
for(int i = 1;i <= m;i ++) {
s = read();o = read();k = read();
ins(s,o,k);
}
for(int i = 0;i <= n;i ++) {
pr[i] = 0x3f3f3f3f; nw[i] = 1;
dp[i] = -1;
}
dp[1] = 0; pr[1] = 0; nw[1] = 0;
upd(1,1);
int t = 0;
for(int i = 1;i <= n;i ++) {
int tt = tre[1];
if(!tt) break;
if(t && (pr[tt] != pr[t] || nw[tt] != nw[t])) rk[tt] = rk[t] + 1;
else rk[tt] = rk[t];
t = tt;
for(int j = hd[t];j;j = nx[j]) {
int y = v[j],z = w[j];
if(rk[t] < pr[y] || (rk[t]==pr[y] && z < nw[y])) {
pr[y] = rk[t]; nw[y] = z;
dp[y] = (dp[t]<<1|z) % MOD;
upd(y,y);
}
}
upd(t,0);
}
for(int i = 2;i <= n;i ++) {
AIput(dp[i],i==n ? '\n':' ');
}
return 0;
}

【NOI P模拟赛】寻找道路(bfs,最短路)的更多相关文章

  1. [NOI P模拟赛] 传统艺能(子序列自动机、矩阵乘法,线段树)

    (2:00)OID:"完了,蓝屏了!"(代码全消失) 众人欢呼 OID:开机,"原题测试--" (30min later)OID 开始传统艺能: " ...

  2. luogu2296 [NOIp2014]寻找道路 (bfs)

    反着建边,从T bfs找合法的点,然后再正着bfs一下求最短路就行了 #include<bits/stdc++.h> #define pa pair<int,int> #def ...

  3. noip模拟赛 保留道路

    [问题描述] 很久很久以前有一个国家,这个国家有N个城市,城市由1,2,3,…,N标号,城市间有M条双向道路,每条道路都有两个属性g和s,两个城市间可能有多条道路,并且可能存在将某一城市与其自身连接起 ...

  4. 洛谷 P2296 寻找道路 —— bfs

    题目:https://www.luogu.org/problemnew/show/P2296 第一次用 Emacs 对拍,写了半天: 注意那个 is 赋值的地方很容易错,千万别反复赋值: 一道水题写了 ...

  5. 【NOI P模拟赛】最短路(树形DP,树的直径)

    题面 给定一棵 n n n 个结点的无根树,每条边的边权均为 1 1 1 . 树上标记有 m m m 个互不相同的关键点,小 A \tt A A 会在这 m m m 个点中等概率随机地选择 k k k ...

  6. NOIP模拟赛 寻找

    题目描述 “我有个愿望,我希望穿越一切找到你.” 这是个二维平面世界,平面上有n个特殊的果实,我从(0,0)点出发,希望得到尽量多的果实,但是出于某种特殊的原因,我的运动方式只有三种(假设当前我在(x ...

  7. 模拟赛20181015 Uva1078 bfs+四维dp

    题意:一张网格图,多组数据,输入n,m,sx,sy,tx,ty大小,起终点 接下来共有2n-1行,奇数行有m-1个数,表示横向的边权,偶数行有m个数,表示纵向的边权 样例输入: 4  4  1  1  ...

  8. noi.acNOIP模拟赛5-count

    题目链接 戳我 题意简述 你有一个n+1个数的序列,都是1~n,其中只有一个有重复,求每个长度的本质不同的子序列个数.\(mod 1e9+7\). sol 说起来也很简单,设相同的数出现的位置为\(l ...

  9. 【XJOI】【NOI考前模拟赛7】

    DP+卡常数+高精度/  计算几何+二分+判区间交/  凸包 首先感谢徐老师的慷慨,让蒟蒻有幸膜拜了学军的神题.祝NOI2015圆满成功 同时膜拜碾压了蒟蒻的众神QAQ 填填填 我的DP比较逗比……( ...

随机推荐

  1. Xmind演讲主题大纲

    参考:https://www.bilibili.com/video/BV1Rb411s7VG?p=9

  2. Dev C++编写C/C++程序 出现[Error] ld returned 1 exit status报错分析及解决

    debug系列第一弹,不知道大家写程序的时候是不是都遇到过如题的报错. 我本人是经常遇到这行熟悉的令人不知所措的报错,可能是我太笨了 有时候百度无果也差不到原因,那就汇总一下目前我遇到的情况吧--持续 ...

  3. Linux服务器安装图形化界面

    Linux服务器安装图形化界面 1.检查有无安装gnome桌面 [root@localhost ~]# rpm -qa |grep gnome 2.查看可安装组件列表 [root@localhost ...

  4. ExtJS 布局-Fit布局(Fit Layout)

    更新记录: 2022年5月31日 第一稿. 1.说明 Fit布局只会显示一个子组件,子项组件的尺寸会拉伸到容器的尺寸.当容器进行调整大小(resized),子组件会自动调整去拉伸到付容器的大小. 注意 ...

  5. Jmeter(五十四) - 从入门到精通高级篇 - 如何在linux系统下运行jmeter脚本 - 上篇(详解教程)

    1.简介 上一篇宏哥已经介绍了如何在Linux系统中安装Jmeter,想必各位小伙伴都已经在Linux服务器或者虚拟机上已经实践并且都已经成功安装好了,那么今天宏哥就来介绍一下如何在Linux系统下运 ...

  6. .Net Core 企业微信更新模版卡片消息

    1.搭建回调服务器 可参考:https://www.cnblogs.com/zspwf/p/16381643.html进行搭建 2.编写代码 2.1接口定义 应用可以发送模板卡片消息,发送之后可再通过 ...

  7. BluePrism手把手教程2.0 创建流程

    2.0.1 创建流程 2.0.2 设置流程名称 2.0.3 添加流程说明 2.0.4 添加流程成功 2.0.4 打开新建的流程 RPA行业微信交流群,欢迎大家扫码加入一起交流,此群用于RPA行业技术. ...

  8. this关键字、static关键字、方法的调用

    1.带有static关键字的方法,不可使用this关键字.因为其调用方法为类名.方法名(建议这种方式,调用不需要对象的参与),不存在对象. 2.实例方法调用必须有对象的存在,先创建对象,通过引用.的方 ...

  9. 一网打尽异步神器CompletableFuture

    最近一直畅游在RocketMQ的源码中,发现在RocketMQ中很多地方都使用到了CompletableFuture,所以今天就跟大家来聊一聊JDK1.8提供的异步神器CompletableFutur ...

  10. [ 1 x 1 ] Convolution-1*1卷积的作用

    一.卷积神经网络中的卷积(Convolution in a convoluted neural network) 具体内容亲参考<深度学习>. 二.1*1卷积(one by one con ...