题目背景

在北纬 91° ,有一个神奇的国度,叫做企鹅国。这里的企鹅也有自己发达的文明,称为企鹅文明。因为企鹅只有黑白两种颜色,所以他们的数学也是以二进制为基础发展的。

比如早在 111010011110100111101001 年前,他们就有了异或这样一个数学概念。如果你不知道异或是什么,请出门过墙左转到这里

再比如早在 100001010000101000010 年前,他们的大科学家 Penguin. Tu 就提出了最短路径这样一些概念。

题目描述

企鹅国中有 NNN 座城市,编号从 111 到 NNN 。

对于任意的两座城市 iii 和 jjj ,企鹅们可以花费 (i xor j)×C(i~\mathrm{xor}~j) \times C(i xor j)×C 的时间从城市 iii 走到城市 jjj ,这里 CCC 为一个给定的常数。

当然除此之外还有 MMM 条单向的快捷通道,第 iii 条快捷通道从第 FiF_iFi​​​ 个城市通向第 TiT_iTi​​​ 个城市,走这条通道需要消耗 ViV_iVi​​​ 的时间。

现在来自 Penguin Kingdom University SDFZ 的企鹅豆豆正在考虑从城市 AAA 前往城市 BBB 最少需要多少时间?

输入输出格式

输入格式:

从标准输入读入数据。

输入第一行包含三个整数 N,M,C,表示企鹅国城市的个数、快捷通道的个数以及题面中提到的给定的常数C。

接下来的 MMM 行,每行三个正整数 Fi,Ti,Vi (1≤Fi≤N1,1≤Ti​≤N,1≤Vi​≤100),分别表示对应通道的起点城市标号、终点城市标号和通过这条通道需要消耗的时间。

最后一行两个正整数 A,BA,BA,B (1≤C≤100),表示企鹅豆豆选择的起点城市标号和终点城市标号。

输出格式:

输出到标准输出。

输出一行一个整数,表示从城市 A 前往城市 B 需要的最少时间。

题目分析

关于SPFA:它死了。讲真我不觉得在类似网格图的东西上用SPFA是什么好办法

但即使是Dijkstra也完全无法承受如此巨大的数据范围,所以一定不是暴力建边。

折腾一下可以发现一些神奇性质:我们可以试着把这些点转成二进制,与其一下把所有位都转化成需要的数字不如一位一位转化,反正边权和是一样的。

举个例子,从00000到11111,直接一步到边权位00000 xor 11111 = 11111,我们也可以00000 -> 00001 -> 00011 -> 00111 -> 01111 -> 11111,发现这样和原来的代价是一样的!

根据这样的性质,我们只要把x和x*2^k连边就可以了(这里读作“二的k次方”,“^”不表示xor)。

再加上玄学优化的Dijkstra就可以了,堆优化好像有点悬,得吸氧

Code

// luogu-judger-enable-o2
#include<iostream>
#include<cstdio>
#include<queue>
#include<algorithm>
#include<cctype>
using namespace std; const int MAXN = + ;
const int MAXM = + ;
const int INF = 0x3f3f3f3f; inline int read() {
int X = ,w = ;char ch = ;
while(!isdigit(ch)) {w |= ch == '-';ch = getchar();}
while(isdigit(ch)) X = (X<<)+(X<<)+(ch^),ch = getchar();
return w ? -X : X;
} struct Edge {
int nxt;
int to,w;
} l[MAXM + MAXN * ];
struct Node {
int id;
int dis;
friend bool operator < (Node x,Node y) {
return x.dis > y.dis;
}
} a[MAXN]; int n,m,c;
int s,t;
int head[MAXN],cnt;
int dis[MAXN];
bool vis[MAXN];
//priority_queue <pair,vector<pair>,greater<pair>> q;
priority_queue<Node> q; void add(int x,int y,int z) {
cnt++;
l[cnt].nxt = head[x];
l[cnt].to = y;
l[cnt].w = z;
head[x] = cnt;
return;
} inline void dijkstra(int s) {
// int cur;
// memset(dis,127,sizeof(dis));
// dis[s] = 0;
// q.push(pair(0,s));
// while(!q.empty()) {
// cur = q.top().second;
// tmp = q.top().first;
// q.pop();
// if(tmp != dis[cur]) continue;
// for(int i = head[cur];i;i = l[i].nxt){
// if(dis[l[i].to] > dis[cur] + l[i].w) {
// dis[l[i].to] > dis[cur] + l[i].w
// q.push(pair(dis[l[i].to],l[i].to));
// }
// }
// }
// return;
for(int i = ;i <= n;i++) {
a[i].dis = INF;
a[i].id = i;
}
int cur;
a[s].dis = ;
q.push(a[s]);
while(!q.empty()) {
cur = q.top().id;
q.pop();
if(vis[cur]) continue;
vis[cur] = true;
for(int i = head[cur];i;i = l[i].nxt) {
if(vis[l[i].to]) continue;
if(a[l[i].to].dis > a[cur].dis + l[i].w) {
a[l[i].to].dis = a[cur].dis + l[i].w;
q.push(a[l[i].to]);
}
}
}
} int main() {
scanf("%d%d%d",&n,&m,&c);
int x,y,z;
for(register int i = ;i <= m;i++) {
x=read(),y=read(),z=read();
add(x,y,z);
}
for(register int i = ;i <= n;i++) {
for(register int j = ;j <= n;j <<= ) {
if((i ^ j) > n) continue;
add(i,i^j,j*c);
}
}
// for(int i = 1;i <= cnt;i++) {//DEBUG
// cout<<"DEBUG: "<<i<<" "<<l[i].nxt<<" "<<l[i].to<<" "<<l[i].w<<endl;
// }
scanf("%d%d",&s,&t);
dijkstra(s);
printf("%d\n",a[t].dis);
// for(int i = 1;i <= n;i++) cout<<a[i].dis<<" ";
return ;
}

[Luogu] P4366 [Code+#4]最短路的更多相关文章

  1. luogu 4366 [Code+#4]最短路 Dijkstra + 位运算 + 思维

    这个题思路十分巧妙,感觉很多题都有类似的套路. 我们发现异或操作其实就是将一个数的二进制的若干个 $0$ 变成 $1$,或者一些 $1$ 变成 $0$. 而每次按照某种顺序一位一位地异或也可以起到同时 ...

  2. [Code+#4]最短路 (最短路)

    [Code+#4]最短路 题目背景 在北纬 91° ,有一个神奇的国度,叫做企鹅国.这里的企鹅也有自己发达的文明,称为企鹅文明.因为企鹅只有黑白两种颜色,所以他们的数学也是以二进制为基础发展的. 比如 ...

  3. [Luogu P3953] 逛公园 (最短路+拓扑排序+DP)

    题面 传送门:https://www.luogu.org/problemnew/show/P3953 Solution 这是一道神题 首先,我们不妨想一下K=0,即求最短路方案数的部分分. 我们很容易 ...

  4. [Code+#4]最短路 解题报告

    Luogu · 传送门 Orz THU众大佬,lct(注意不是link-cut-tree,是一个大佬) 这道题很容易让人联想到 最短路,但是最短路需要先 建图: 暴力建出所有边的算法显然是不可行的,因 ...

  5. LOJ6354 & 洛谷4366:[Code+#4]最短路——题解

    https://loj.ac/problem/6354 https://www.luogu.org/problemnew/show/P4366 题面见上面. 这题很妙,且可能是我傻,感觉这题不太好想. ...

  6. luogu 1608 路径统计--最短路计数

    https://www.luogu.org/problemnew/show/P1608 题意https://www.cnblogs.com/rmy020718/p/9440588.html相似,建议还 ...

  7. Luogu P3489 [POI2009]WIE-Hexer 最短路

    https://www.luogu.org/problemnew/show/P3489 普通的最短路,不过我觉得这个复杂度按道理来说边数不应该是m*2^13吗,不知道是数据比较水还是实际上能证明复杂度 ...

  8. luogu【P1144】最短路计数

    原题入口 这道题 一道有关于最短路的图论问题. 要求从1开始求解最短路的条数. 这个题十分有趣,首先,跑裸的spfa(或者dijkstra)算出从1开始的最短路的长度. 再其次,计数的话,可以用记忆化 ...

  9. luoguP4366 [Code+#4]最短路 最短路

    好久没写过博客了.... 本题还是挺有趣的(很水的最短路) 关键在于怎么优化这$n^2$条连边 通常,我们希望用一些边来替代一条边从而减小边集 那么,注意到异或操作可以拆分成按位运算,因此我们只需考虑 ...

随机推荐

  1. dom小练习

    dom小练习 学习要点 综合运用学过的知识完成几个综合小练习,巩固学过的知识. 阶段小练习8-1:改变网页字体的大小 要求和提示: 要求:当用户选择‘大/中/小’的选项时,页面字体发生相应的变化 阶段 ...

  2. luogu 2622 关灯问题II

    题目大意: 有一些灯,有些开关可以控制这些灯,给出矩阵表示控制 对于矩阵中的a i j 表示第i个开关控制第j个灯的情况 若元素为1 表示当灯开着的时候,关掉灯 若元素为0 表示无操作 若元素为-1 ...

  3. SYSUCPC2017 DAG最多能添加多少边?

    校赛的一道题目 难度不大 不过还是挺有趣的 题意:给定一个有向图,问此图是不是一个DAG且不包含重边 如果是的话 回答最多可以添加多少条边且图仍然是个DAG 考虑对于任意一个点u 添加一条边(u,v) ...

  4. 杂项-Java:Tomcat

    ylbtech-杂项-Java:Tomcat 1.返回顶部 1. Tomcat是Apache 软件基金会(Apache Software Foundation)的Jakarta 项目中的一个核心项目, ...

  5. Spring中bean的作用域与生命周期

    在 Spring 中,那些组成应用程序的主体及由 Spring IOC 容器所管理的对象,被称之为 bean.简单地讲,bean 就是由 IOC 容器初始化.装配及管理的对象,除此之外,bean 就与 ...

  6. 怎样去思考问题 解决问题 zkc学长的福利

    题目描述 又是一年ACM集训的时刻,zkc学长邀请n位同学来玩一个有奖游戏.首先,他让每个同学在左.右手上面分别写下一个整数,zkc学长自己也在左.右手上各写一个整数.然后,让这n位同学排成一排,zk ...

  7. [LOJ#10064]黑暗城堡

    Description 在顺利攻破 Lord lsp 的防线之后,lqr 一行人来到了 Lord lsp 的城堡下方.Lord lsp 黑化之后虽然拥有了强大的超能力,能够用意念力制造建筑物,但是智商 ...

  8. 数学 HDOJ 5301 Buildings

    题目传送门 /* 题意:n*m列的矩阵,删除一个格子x,y.用矩形来填充矩阵.且矩形至少有一边是在矩阵的边缘上. 求满足条件的矩形填充方式中面积最大的矩形,要使得该最大矩形的面积最小. 分析:任何矩形 ...

  9. Shape Drawable Resources

    1,示例 它们的代码如下: shape_oval.xml <?xml version="1.0" encoding="utf-8"?> <sh ...

  10. hbase优化小结

    目录: 1,背景 2,GC 3,hbase cache 4,compaction 5,其他 1,背景 项目组中,hbase主要用来备份mysql数据库中的表.主要通过接入mysql binlog,经s ...