考试的时候写了个dfs找出来了,最后处理的时候想到了贪心,但是正确性没有想通。然后想了想动规,也没想通。最后没办法,用状态的话用了个状压,弄了40分。

正解是bfs+贪心。Dfs也有过的。

下面题解引用自他人:

整体的思路是这样的:

从第一层的每一个点开始往下做一次搜索,每一次搜索可以覆盖到最后一行的一段,我们就可以把它看成是一条线段;

做完搜索之后我们就得到了一坨线段。。然后这就变成了一个线段覆盖问题,用最少的线段来把最后一行盖满,当然也存在无论如何都盖不满的情况。。

下面来说说线段覆盖的思路:

把所有的线段按照左端点排序,之后先以第一条线段为基准,第一条线段能覆盖到的地方就是我们现在能覆盖到的区域(下面成为灰色区域);然后从左往右开始扫,找一条左端点在灰色区域内,而右端点伸出灰色区域最长的线段,就取这条线端,更新灰色区域,一直找扫完最后一条线段为止,贪心就能求出来最小的线段数;

其实有一个小小的优化:

关于第一行,其实是不必要每一个点做一次搜索的,我们发现只有在这个点的高度>=它左边的点而且又>=它右边的点这种情况下才需要做搜索,因为如果该点的高度比它左右两边任意一个点低的话,做那个点的时候就已经把它给搜索过了,是冗余的,所以没必要;

注意(我犯的错误):

①  一定要一个点做一次独立的搜索,切忌把所有的一起搜索,那样会T掉。用bfs的话也会内存超限

②  搜索的时候要打标记免得重复,不然也会T掉

证明如果有解线段一定是连续的:

可能会想到有没有拐着弯线段不连续的情况。这种情况其实就是无解的情况。因为如果中间存在空白,说明上面下不来,那么所有的水都下不来。

代码有注解:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<queue>
#include<algorithm>
using namespace std;
#define pos(i,a,b) for(int i=(a);i<=(b);i++)
#define pos2(i,a,b) for(int i=(a);i>=(b);i--)
#define N 510
int n,m;
struct haha
{
    int fir;
    int hang,lie;
};
struct qian
{
    int from,to;
    qian(){from=0x7fffffff;}
}cun[N];
int hei[N][N];
int flag[N];
int biao[N][N];
void bfs()
{
     pos(i,1,m)
     {
         if(hei[1][i]>=hei[1][i-1]&&hei[1][i]>=hei[1][i+1])
         {
            queue<haha> q;
            memset(biao,0,sizeof(biao));
            haha tmp;
            tmp.hang=1;
            tmp.lie=i;
            tmp.fir=i;
            q.push(tmp);
            while(!q.empty())
            {
                 haha tmp=q.front();
                 if(tmp.hang==n)
                 {
                   cun[tmp.fir].from=min(cun[tmp.fir].from,tmp.lie);
                   cun[tmp.fir].to=max(cun[tmp.fir].to,tmp.lie);
                   flag[tmp.lie]=1;
                 }
                 if(tmp.lie-1>0&&hei[tmp.hang][tmp.lie]>hei[tmp.hang][tmp.lie-1]&&biao[tmp.hang][tmp.lie-1]==0)
                 {
                    haha tmp2=tmp;
                    tmp2.lie--;
                    biao[tmp.hang][tmp.lie-1]=1;
                    q.push(tmp2);
                 }
                 if(tmp.lie+1<=m&&hei[tmp.hang][tmp.lie]>hei[tmp.hang][tmp.lie+1]&&biao[tmp.hang][tmp.lie+1]==0)
                 {
                    haha tmp2=tmp;
                    tmp2.lie++;
                    biao[tmp.hang][tmp.lie+1]=1;
                    q.push(tmp2);
                 }
                 if(tmp.hang+1<=n&&hei[tmp.hang][tmp.lie]>hei[tmp.hang+1][tmp.lie]&&biao[tmp.hang+1][tmp.lie]==0)
                 {
                    haha tmp2=tmp;
                    tmp2.hang++;
                    biao[tmp.hang+1][tmp.lie]=1;
                    q.push(tmp2);
                 }
                 if(tmp.hang-1>0&&hei[tmp.hang][tmp.lie]>hei[tmp.hang-1][tmp.lie]&&biao[tmp.hang-1][tmp.lie]==0)
                 {
                    haha tmp2=tmp;
                    tmp2.hang--;
                    biao[tmp.hang-1][tmp.lie]=1;
                    q.push(tmp2);
                 }
                 q.pop();
             }
         }
     }
}
int ans;
bool aaa(const qian &a,const qian &b)
{
     return a.from<b.from;
}
int main()
{
    scanf("%d%d",&n,&m);
    pos(i,1,n)
      pos(j,1,m)
        scanf("%d",&hei[i][j]);
    bfs();
    pos(i,1,m)
    {
       if(flag[i]==0)
         ans++;
    }//如果没有覆盖的情况,输出即可
    if(ans)
    {
       printf("0\n%d",ans);
       return 0;
    }
    sort(cun+1,cun+m+1,aaa);//排序左端点
    int cnt;
    pos(i,1,m)//找出线段的个数
    {
       if(cun[i].from==0x7fffffff)
       {
         cnt=i-1;
         break;
       }
    }
    int you=1,Ma=0;
    while(1)//贪心答案
    {
      pos(j,1,cnt)
      {
          if(cun[j].from<=you)
             Ma=max(Ma,cun[j].to);//找出左端点已选择区域内右端点最长值
          else
            break;
      }
      you=Ma+1;Ma=0;
      ans++;
      if(you>m)
        break;
    }
    printf("1\n%d",ans);
    return 0;
}

  

[NOIP 2010]饮水入城 搜索+贪心的更多相关文章

  1. [NOIP 2010] 引水入城

    搜索+贪心. 参考博客:http://blog.sina.com.cn/s/blog_8442ec3b0100xib1.html 主要是要看出来,如果有解的话,每个沿湖城市能够流到的范围是连续的区间. ...

  2. noip 2010 引水入城 贪心 + 搜索

    不难分析出如果有解则每个蓄水厂所能覆盖到的干旱城市一定是连续的.否则,中间那些没被覆盖的部分永远都不能被覆盖到. 当然,每个蓄水厂所覆盖的城市有可能不连续,不过既然有解,则一定都是连续的.我们可以开一 ...

  3. [NOIP 2010] 引入入城

    [题目链接] https://loj.ac/problem/2595 [算法] 显然 , 每个第一行的成市控制的一定是一段区间 那么 , 问题就转化为了经典的区间覆盖问题 , 贪心即可 , 时间复杂度 ...

  4. 洛谷P1514 引水入城 [搜索,区间DP]

    题目传送门 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 N 行×M 列的矩形,如上图所示,其中每个格子都代表一座城市,每 ...

  5. vijos p1777 引水入城(bfs+贪心)

    引水入城   描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N行M列的矩形,其中每个格子都代表一座城市,每座城市都有一个海拔高度. 为了使 ...

  6. [luogu]P1514 引水入城[搜索][记忆化][DP]

    [luogu]P1514 引水入城 引水入城 题目描述在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形 ,如下图所示,其中每个格 ...

  7. NOIP2010_T4_引水入城 bfs+贪心

    在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 N 行 M 列的矩形,如上图所示,其中每个格子都代表一座城 市,每座城市都有一个海拔高度.为了使 ...

  8. luogu 1066 引水入城(bfs+贪心)

    90分,有一个点TLE.... 首先可以证明一个东西,如果从上面一排的某个点bfs一次到最下面一排的饮水点不是一个区间的话,那么最后一定所有饮水点不会被覆盖完的. 证明考虑反证法. 所以从上面一排的每 ...

  9. noip2010 引水入城 bfs+贪心

    如果能够实现,每个河边的城市对应的控制区域一定是一条线段. 所以直接bfs每个河边的城市,贪心线段的右端点 #include<cstdio> #include<cstring> ...

随机推荐

  1. hive集成sentry的sql使用语法

    Sentry权限控制通过Beeline(Hiveserver2 SQL 命令行接口)输入Grant 和 Revoke语句来配置.语法跟现在的一些主流的关系数据库很相似.需要注意的是:当sentry服务 ...

  2. 【Android Developers Training】 4. 启动另一个Activity

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  3. Python序列类型

    Python序列类型 序列:字符.列表.元组 所有序列都支持迭代 序列表示索引为非负整数的有序对象集合 字符和元组属于不可变序列,列表可变 1)字符 字符串字面量:把文本放入单引号.双引号或三引号中: ...

  4. MyBatis基础

    一.大配置 <?xml version="1.0" encoding="UTF-8" ?> <!--文档类型,头文件--> <!D ...

  5. webpack初步介绍

    我们通过npm -g可以安装一个webpack的东西. npm -g叫做全局安装,通常是安装CLI程序(commond line interface). 我们只用过一次,装了cnpm.此时就能在CMD ...

  6. dfs.datanode.max.transfer.threads

    An HDFS DataNode has an upper bound on the number of files that it will serve at any one time: <p ...

  7. 关于Lumen / Laravel .env 文件中的环境变量是如何生效的

    .env 文件包含默认环境变量,我们还可自定义其他任何有效的变量,并可通过  调用 env() 或 $_SERVER 或 $_ENV  来获取该变量.那么env()是如何加载到这些变量的呢?在Lume ...

  8. zepto.js 的tap事件中点击一次触发两次事件

    html代码: <div class="xh-lxx-cart-count1"> <span class="minus">-</s ...

  9. cocos2dx 图片压缩工具 推荐

    随着版本的第二次迭代,包体越来越大;随之而来的也就是下载量会有所影响;毕竟每增大一点包体大小,下载的玩家就会损失一小批; 但是我们的图片都经过美术压缩打包,依然包体不小; 最后发现是因为把图片都打包成 ...

  10. c++动态内存管理

    一.内存的简要了解 说到内存,很多人应该都多多少少有点了解了,我们在这再稍微多说几句: 一般我们可以把内存理解为三个部分:静态区,栈,堆.有些朋友搞不清到底什么是栈什么是堆,堆栈有多人会认为是堆和栈, ...