牛客练习赛67 E. 牛妹游历城市

牛妹游历城市

https://ac.nowcoder.com/acm/contest/6885/E

Description

牛妹现在正在1号点(自己家里),他决定前往n号点(牛妹想去的地方),中途可以多次经过1~n号点。
现在,已知每个点都有个权值 ,如果 & ≠0,则i号点和j号点之间连有一条双向边,权值为。他想要最小化自己的行走距离,但是他计算不出来qaq。相信全牛客最聪明的你一定会吧!

Solution

朴素做法是暴力连所有边,但是边集规模过大,考虑二进制优化,对32个二进制位建立32个虚点,对于每个,如果该二进制位j为1,那么连边权值为 1 << j。(注意虚点到该点的距离为0)
最后跑一个Dijkstra即可,注意 ,不能开 int

图片说明

Code

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1e5 + 105;
ll a[N];
bool vis[N];
ll dist[N];
struct Edge {
    int v;
    ll cost;
    Edge(int _v = 0, ll _cost = 0): v(_v), cost(_cost){}
};
vector<Edge> G[N];
struct qnode {
    int v;
    ll cost;
    bool operator < (const qnode &s) const {
        return cost > s.cost;
    }
    qnode(int _v = 0, ll _cost = 0): v(_v), cost(_cost){}
};
void Dijkstra() {
    priority_queue<qnode> q;
    q.push(qnode(1, 0));
    dist[1] = 0;
    while(!q.empty()) {
        qnode now = q.top(); q.pop();
        int u = now.v;
        if(vis[u]) continue;
        vis[u] = true;
        for(int i = 0; i < G[u].size(); i++) {
            int v = G[u][i].v;
            ll cost = G[u][i].cost;
            if(!vis[v] && dist[v] > dist[u] + cost) {
                dist[v] = dist[u] + cost;
                q.push({v, dist[v]});
            }
        }
    }
}
int main() {
    ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
    int T; cin >> T;
    while(T--) {
        int n; cin >> n;
        for(int i = 1; i <= n; i++) cin >> a[i], G[i].clear(), dist[i] = 1e18, vis[i] = false;
        for(int i = n + 1; i <= n + 50; i++) {
            G[i].clear(), dist[i] = 1e18, vis[i] = false;
        }
        for(int i = 1; i <= n; i++) {
            for(int j = 0; j < 32; j++) {
                if(a[i] & (1LL << j)) {
                    G[i].push_back({n + 1 + j, 1LL << j});
                    G[n + 1 + j].push_back({i, 0});
                } 
            }
         }
        Dijkstra();
        if(dist[n] == 1e18) cout << "Impossible\n";
        else cout << dist[n] << "\n";
    }
    return 0;
}
全部评论
感觉跟小雨坐地铁那道差不多
点赞 回复 分享
发布于 2020-08-15 19:40

相关推荐

头像
10-22 20:13
中南大学 Java
序言大家好呀。我是希晨er,一个初入职场的程序猿小登最近上班摸鱼刷到了一篇文章:10年深漂,放弃高薪,回长沙一年有感,还有聊聊30岁大龄程序员过往的心路历程,突然就有点感慨。我如今也做出了和大明哥一样的抉择,只是更早。此外我22年的人生,好像从来没好好记录过。正好现在工作不太忙,就想把这些经历写下来,也希望能得到社区里各位前辈的指点个人背景我是03年出生的西安娃,父母都是普通打工人。刚从中南大学软件工程专业毕业半年,现在在老家的央企过着躺平摆烂的日子成长轨迹从农村到城市的童年我家并不是西安的,只是爸妈在西安上班,从小学之后就把我接到了西安。后来老家房子拆了,爷爷奶奶也搬了过来。农村的生活,我觉...
Yki_:看哭了,恋爱那一段你女朋友说你不够关心她,可你毕竟也愿意遇到矛盾写几千字来和她慢慢分析;说不愿意给她花钱,我感觉可能只是消费观不一样;如果她想留在长沙,也应该提前跟你说开。不过她也许会心疼你放弃大厂offer转向数字马力?我也因为同样的原因有过一段幸福而充满遗憾的感情,不过跟爱情相比确实前途更重要一点。至于offer的选择,换我我也会这么选。把这些旧事记录下来以后,接下来就好好向前看吧,加油兄弟
🍊晨光随笔
点赞 评论 收藏
分享
Aurora23:属于挂一半,暂时进池子了,隔一段时间没有其他组捞的话就彻底结束了
点赞 评论 收藏
分享
评论
7
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务