The 17-th BIT Campus Programming Contest C.小L的旅行

T h e 17 − t h B I T C a m p u s P r o g r a m m i n g C o n t e s t \Huge{The 17-th BIT Campus Programming Contest} The17−thBITCampusProgrammingContest

C . 小 L 的旅行 \huge{C.小L的旅行} C.小L的旅行

题目链接:The 17-th BIT Campus Programming Contest

文章目录

题意

给出一个 n n n个点的有向图,有 m m m条边且每条边的边权为 1 1 1。每个点对应有一个魔法值,对于任意两点,若存在: a i & a j = a j a_i \& a_j = a_j ai&aj=aj,那么就存在一条从 i i i到 j j j的边,边权为 1 1 1。

求从 1 1 1到其他点的最短距离。

思路

题目要求求最短路,因此考虑使用 D i j k s t r a Dijkstra Dijkstra。由于边权都为 1 1 1,我们考虑优化 D i j k s t r a Dijkstra Dijkstra,可以使用双端队列优化掉优先队列。

然后考虑加边(将根据魔法值建的边加入到图中)。

  • 满足 a i & a j = a j a_i \& a_j = a_j ai&aj=aj,则 a j a_j aj是由 a i a_i ai的二进制下去掉若干 1 1 1得到。

  • 求一个数的所有符合 a i & a j = a j a_i \& a_j = a_j ai&aj=aj条件的值的时间复杂度太大

  • 考虑只将 a i a_i ai与其二进制下某一位1变为0得到的数之间建边

  • 这样 a i a_i ai将会到达所有能够建边的数,虽然不是强连通,但是边权都为 0 0 0。

标程

cpp 复制代码
#define int long long 
const int INF = 0x7fffffff;
const int N = 2e6 + 10; 

vector<int> d(N), dis(N, INF);
vector<PII> g[N];
vector<bool> vis(N);

void Solved() {
    int n, m; cin >> n >> m;
    for(int i = 1; i <= n; i ++ ) cin >> d[i];

    for(int i = 1; i <= m; i ++ ) {
        int x, y; cin >> x >> y;
        g[x].push_back({y, 1});
    }
    for(int i = 0; i < (1 << 20); i ++ )	//预处理出根据魔法值建的边(建在新图中)
        for(int j = 0; j < 20; j ++ )
            if((i >> j) & 1)
                g[n + i + 1].push_back({n + 1 + (i ^ (1 << j)), 0});

    for(int i = 1; i <= n; i ++ ) {			//合并两图
        g[i].push_back({n + 1 + d[i], 1});	//代表使用魔法,边权为1
        g[n + 1 + d[i]].push_back({i, 0});	//能进入新图则表示边权已+1,此时不需要加边权
    }
    
    dis[1] = 0;
    deque<int> dq;		//双端队列优化
    dq.push_back(1);
    while(dq.size()) {
        int u = dq.front(); dq.pop_front();	//每次取队列头元素
        if(vis[u]) continue;
        vis[u] = true;
        for(auto [i, j] : g[u]) {
            if(dis[i] <= dis[u] + j) continue;

            dis[i] = dis[u] + j;
            if(j) dq.push_back(i);	//优化的具体方法,及边权为1则放在末尾
            else dq.push_front(i);	//边权为0则放在开头
        }
    }

    for(int i = 1; i <= n; i ++ ) {
        if(dis[i] != INF) {
            cout << dis[i] << endl;
        } else {
            cout << "-1\n";
        }
    }
}
相关推荐
千弥霜7 分钟前
codeforces1914 C~F
c语言·算法
wyiyiyi11 分钟前
【数据结构+算法】进栈顺序推算、卡特兰数与逆波兰表达式
汇编·数据结构·笔记·算法
天若有情67314 分钟前
Multi-Stride Predictive RNG:革命性的可控随机数生成算法
算法·算法设计·c++编程·随机数生成·msp-rng·魔术算法
C_Liu_24 分钟前
14:C++:二叉搜索树
算法
CC-NX34 分钟前
32位汇编:实验9分支程序结构使用
汇编·算法·win32·分支结构
万岳科技系统开发40 分钟前
外卖小程序中的高并发处理:如何应对大流量订单的挑战
算法·小程序·开源
TL滕43 分钟前
从0开始学算法——第二天(时间、空间复杂度)
数据结构·笔记·学习·算法
旺仔老馒头.2 小时前
【数据结构与算法】手撕排序算法(二)
c语言·数据结构·算法·排序算法
好学且牛逼的马3 小时前
【Hot100 | 2 LeetCode49 字母异位词分组问题】
算法
2301_795167203 小时前
Rust 在内存安全方面的设计方案的核心思想是“共享不可变,可变不共享”
算法·安全·rust