【题目链接】
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 <iostream>
#include <algorithm>
#include <stdio.h>
#include <string.h>
#include <queue>
#include <string>
#include <map>
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_queue<node>q;
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;
}