【hdu 5418】 Victor and world

【题目链接】

          点击打开链接

【算法】

         状压DP

         f[i][S]表示走的最后一步在i,状态为S

         于是我们可以用最短路径 + 状压DP解决此题,由于不存在负边,所以可以用dijkstra+堆优化


【代码】

        

#include<bits/stdc++.h>
using namespace std;
#define MAXN 16

struct info {
    int home,s,dis;    
    friend bool operator < (info a,info b) {
        return a.dis > b.dis;
    }
};

int T,N,M,i,j,k,mask,u,v,w;
int dist[MAXN+10][MAXN+10],f[MAXN+10][(1<<MAXN)+10],vis[MAXN+10][(1<<MAXN)+10];
priority_queue<info> q;
info x;

template <typename T> inline void read(T &x) {
    int f = 1; x = 0;
    char c = getchar();
    for (; !isdigit(c); c = getchar()) { if (c == '-') f = -f; }
    for (; isdigit(c); c = getchar()) x = x * 10 + c - '0';
    x *= f;
}
template <typename T> inline void write(T x) {
    if (x < 0) { putchar('-'); x = -x; }
    if (x > 9) write(x/10);
    putchar(x%10+'0');    
}
template <typename T> inline void writeln(T x) {
    write(x);
    puts("");    
}

int main() {
    
    read(T);
    
    while (T--) {
        memset(vis,0,sizeof(vis));
        read(N); read(M);
        for (i = 1; i <= N; i++) {
            for (j = 1; j <= N; j++) {
                dist[i][j] = 2e9;
            }
        } 
        for (i = 1; i <= M; i++) {
            read(u); read(v); read(w);
            dist[u][v] = min(dist[u][v],w);
            dist[v][u] = min(dist[v][u],w);
        }    
        mask = (1 << N) - 1;
        for (i = 1; i <= N; i++) {
            for (j = 1; j <= mask; j++) {
                f[i][j] = 2e9;
            }
        }
        f[1][0] = 0;
        q.push((info){1,0,0});
        while (!q.empty()) {
            x = q.top(); q.pop();
            if (vis[x.home][x.s]) continue;
            vis[x.home][x.s] = 1;
            for (i = 1; i <= N; i++) {
                if (x.dis + dist[x.home][i] < f[i][x.s|(1<<(i-1))]) {
                    f[i][x.s|(1<<(i-1))] = x.dis + dist[x.home][i];
                    q.push((info){i,x.s|(1<<(i-1)),f[i][x.s|(1<<(i-1))]});    
                }
            }
        }
        if (N == 1) writeln(0);
        else writeln(f[1][(1<<N)-1]);
    }    
    
    return 0;
    
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值