HDU 4370 0 or 1(最短路djs)

【题目链接】
http://poj.org/problem?id=4370

题目意思

给你一个矩阵,让你找到一个由0和1组成的矩阵,满足
1.X12+X13+…X1n=1
2.X1n+X2n+…Xn-1n=1
3.∑Xki (1<=k<=n)=∑Xij (1<=j<=n).

解题思路

迷一样的题目意思。总的就是把矩阵看做图,那么条件1就可以看出节点1的出度为1,条件2看做节点n的入点为1,最后就是条件3看做2~n-1的出度等于入度。为了满足上面情况有:
1.有一条从节点1到n的最短路。
2.节点1找一个除了自环的环,节点n找个除了自环的环。两环的总和。
上面两种情况中小的一种就是满足条件的最小值。

代码部分


///代码改前一题的代码而成所以有点丑(虽然本身代码就丑o(╥﹏╥)o)
#include 
#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;
#define LL long long
#define inf 0x3f3f3f3
const int N = 305;
int n,m;
int maps[N][N]; 
int dis[N];
int vis[N];
struct node 
{
    int i,dis;
    friend bool operator < (node a,node b)  
    {
        return a.dis > b.dis;
    }
};
node add(int i,int dis)
{
    node t;
    t.i = i;
    t.dis = dis;
    return t;
}
void Dijkstra(int s)
{    
    priority_queueq;
    for (int i =1; i <= n; i++)
    {
        if (i == s)
            dis[i] = inf;
        else 
        {
            dis[i] = maps[s][i];
            q.push(add(i,dis[i]));
        }
        vis[i] = 0;
    }
    while (!q.empty())
    { 
        node t = q.top();
        q.pop();
        if (vis[t.i])  ///节点i松弛过 
            continue;
        vis[t.i] = 1;
        for (int i = 1; i <= n; i++)
        {
            if (dis[i] > dis[t.i]+maps[t.i][i] && !vis[i])  ///松弛 
            {
                dis[i] = dis[t.i]+maps[t.i][i];
                q.push(add(i,dis[i]));
            }
        }
    }
}
int main()
{
    while (scanf("%d",&n)!=EOF)
    {
        for (int i = 1; i <= n; i++)
        {
            for (int j = 1; j <= n; j++)
                scanf("%d",&maps[i][j]);
        }
        Dijkstra(1);
        int ans = dis[n];
        int res = dis[1];
        Dijkstra(n);
        ans = min(ans,res+dis[n]);
        printf("%d\n",ans);
    }
    return 0;
}

你可能感兴趣的:(最短路)