POJ3592 Instantaneous Transference 强连通+最长路

题目链接:

poj3592



题意:

给出一幅n X m的二维地图,每个格子可能是矿区,障碍,或者传送点 用不同的字符表示;

有一辆矿车从地图的左上角(0,0)出发,只能往右走或往下走,或者通过传送点  选择是否 传送到特定地点

采过的矿的格子 矿会消失;问这辆矿车最多能采多少矿



解题思路:

首先重新建图,将图中二维的顶点压缩成一维的顶点             (方便Tarjan算法)

每个顶点往右,下的顶点建边,传送点的格子往特定顶点建边(建边的两端不能有障碍)

得到一幅可能存在环的有向图;

因为采过矿的格子不可以二次采矿,所以经过某个环等于采集了整个环中所有的矿

我们用Tarjan算法缩点 再重新建图

得到一幅无环有向图,而图中每条边(u->v)的权值应等于value[v]

最后再用spfa求这幅图的最长路即可



代码:

#include
#include
#include
#include
#define maxn 1650
using namespace std;
struct node
{
    int to,next,w;
} edge1[maxn*3],edge2[maxn*3];
int head1[maxn],head2[maxn];
int s1,s2;

int dfn[maxn], low[maxn],num;

int sta[maxn],insta[maxn], top;

int belong[maxn],block;

int n,m,ss,v1[maxn],v2[maxn];
char map[45][45];

void init()
{
    memset(head1,-1,sizeof(head1));
    memset(head2,-1,sizeof(head2));
    memset(dfn,0,sizeof(dfn));
    memset(insta,0,sizeof(insta));
    memset(belong,0,sizeof(belong));
    memset(v2,0,sizeof(v2));
    s1=s2=num=top=block=0;
}

int judge(int x,int y)
{
    if(x<0||y<0||x>=n||y>=m)
        return -1;
    if(map[x][y]>='0'&&map[x][y]<='9')
        return 1;
    if(map[x][y]=='*')
        return 2;
    if(map[x][y]=='#')
        return -1;
}

void addedge(int d,int u,int v,int w)
{
    if(d==1){
        edge1[s1]={v,head1[u]};
        head1[u]=s1++;
    }
    else {
        edge2[s2]={v,head2[u],w};
        head2[u]=s2++;
    }
}

void Tarjan(int u,int pre)
{
    dfn[u]=low[u]=++num;
    insta[u]=1;
    sta[top++]=u;
    for(int i=head1[u];i!=-1;i=edge1[i].next)
    {
        int v=edge1[i].to;
        if(!dfn[v])
        {
            Tarjan(v,u);
            low[u]=min(low[u],low[v]);
        }
        else if(insta[v])
            low[u]=min(low[u],dfn[v]);
    }
    if(dfn[u]==low[u])           //缩点
    {
        block++;
        int d=-1;
        while(d!=u)
        {
            d=sta[--top];
            insta[d]=0;
            belong[d]=block;
            v2[block]+=v1[d];
        }
    }
}

void rebuild()
{
    int u,v;
    for(int i=0;iq;
    int vis[maxn]={0};
    int dis[maxn]={0};
    vis[start]=1;
    dis[start]=v2[start];
    q.push(start);
    while(!q.empty())
    {
        u=q.front();
        q.pop();
        vis[u]=0;
        for(int i=head2[u];i!=-1;i=edge2[i].next)
        {
            v=edge2[i].to;
            if(dis[v]ans)
            ans=dis[i];
    cout<>map[i][j];
                if(map[i][j]=='*')
                    ss++;
            }
        for(int i=1; i<=ss; i++)                //记录第i个传送点的传送位置
            scanf("%d%d",&pos[i][0],&pos[i][1]);

        for(int i=0; i



你可能感兴趣的:(算法)