NOIP2010提高组 引水入城

【问题描述】
这里写图片描述

这道题要分几个成部分看。
第一部分(30分) 一个简单的多源BFS从第一行的点出发能到达哪些最后一行的点。
第二部分 算法1:(60分) 先用多个BFS统计出第一行每个点能到达的最后一行的点,让后枚举每个点选还是不选,最后看能否选完,输出选的最小个数。
算法2:(100分)这种算法你需要发现一个隐藏道具(悄悄告诉你吧:如果能覆盖完,那么第一行每个点所能到的最后一行的点是连续的,有图有真像:真像
如果像图中那样,中间有流不到的点,那么那个部分就会成为孤岛,永远无法到达。)当我们知道这个秘密过后,我们就可以用覆盖区间去做了。
注;还有一个小优化,在考虑第一行的点的时候永远只需要考虑它左右两边的点不高于它的点(自己思考吧)。
下面是满分代码(没加最后一个优化):

#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=505;
struct shu
{
    int x,y;
    friend bool operator <(shu a,shu b)
    {
        return a.x<b.x;
    }
}q[maxn*maxn],f[maxn];
int n,m,a[maxn][maxn];
bool vis[maxn][maxn],d[maxn][maxn]={0},b[maxn]={0};
int dx[]={0,-1,0,1};
int dy[]={1,0,-1,0};

void bfs(int p)
{
    memset(vis,0,sizeof(vis));
    int root=0,frond=0;
    q[root++]=(shu){1,p};
    vis[1][p]=1;
    if(n==1)
    {
    d[p][p]=1;
    b[p]=1;
    }
    while(root!=frond)
    {
        shu t=q[frond++];
        for(int i=0;i<4;i++)
        {
            int xn=t.x+dx[i],yn=t.y+dy[i];
            if(xn<1||xn>n||yn<1||yn>m) continue;
            if(a[xn][yn]>=a[t.x][t.y]) continue;
            if(vis[xn][yn]) continue;
            q[root++]=(shu){xn,yn};
            vis[xn][yn]=1;
            if(xn==n)
            {
            d[p][yn]=1;
            b[yn]=1;
            }
        }
    }

}
int read()
{
    int x=0;
    char ch=getchar();
    while(ch<'0'||ch>'9') ch=getchar();
    while(ch<='9'&&ch>='0')
    {
        x=x*10+ch-'0';
        ch=getchar();
    }
    return x;
}

void init()
{
    n=read();
    m=read();
    for(int i=1;i<=n;i++)
    for(int j=1;j<=m;j++)
    a[i][j]=read();
}
void work()
{
    for(int i=1;i<=m;i++)
    {
        int j;
        for(j=1;j<=m;j++)
        if(d[i][j])
        {
        f[i].x=j;
        break;
        }
        for(;j<=m;j++)
        if(d[i][j]) f[i].y=j;
    }
    int ans=0;
    int s=1,l=m;
    sort(f+1,f+1+m);
    for(int i=1;i<=m;i++)
    if(f[i].x<=s)
    {
        int k=s;
        for(int j=i;f[j].x<=s;j++)
        k=max(k,f[j].y);
        ans++;
        s=k+1;
        if(s>l) break;
    }
    printf("%d\n",ans);
}
int main()
{
    //freopen("in.txt","r",stdin);
    init();
    for(int i=1;i<=m;i++)
    bfs(i);

    int ans=0;
    for(int i=1;i<=m;i++)
    if(b[i]) ans++;
    if(ans<m) printf("0\n%d",m-ans);
    if(ans==m)
    {
        printf("1\n");
        work();
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值