18. 填坑Ⅰ

又是北湖深坑,惊不惊喜,意不意外?! R o a r k 觉得用水填湖太没意思了,用石头填坑多有意思。 假设北湖的地面还是一维的,每一块宽度都为1,高度是非负整数,用一个数组来表示。 现提供不限量的1*2规格的石头,问是否可以将北湖填平。(所有地面到达同一高度即为填平) 注:石头只能水平或垂直填放。

解题思路:

这里提供两种方法,其中第二种的思路可以用到下一题

第一种就是利用二染色,我们先统计出最大的数,然后将最大的数和每一列的长度的差值作为要染色的棋盘,统计染0和染1的个数,因为每一个大小为二的块覆盖都会覆盖一个1和一个0,故如果01个数相同就可以覆盖,如果1比0多一个并且n是奇数,我们就再染一层,可以证明可以覆盖,剩下的情况不能覆盖。

第二种是利用栈

由于我们可以竖着覆盖,因此我们可以把奇数高看成1,偶数高看成0,比他多的便用竖着覆盖填充可以不用考虑。我们先将第一个入栈,再进行遍历,如果遍历到的和栈顶的不一样,如果这个还比栈顶高则返回false,因为栈顶剩下的那个无法覆盖,如果比栈顶低则入栈,如果一样的话就删去栈顶

遍历完后,如果栈中元素大于2则返回false。否则true

为什么栈中元素为1可以呢?

因为如果剩下一个则n必然是奇数,因此可以再填一层将这个奇数消掉

代码:

第一种

#include<iostream>  
using namespace std;  
int a[200010];  
long long b[3];  
int main(){  
    int n;  
    ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);  
    while(cin>>n){  
        int ma=-1;  
        b[0]=0;  
        b[1]=0;  
        for(int i=1;i<=n;i++){  
            cin>>a[i];  
            ma=max(ma,a[i]);  
        }  
        for(int i=1;i<=n;i++){  
            if(i%2){  
                b[1]+=(ma-a[i])/2;  
                b[0]+=((ma-a[i])/2+(ma-a[i])%2);  
            }  
            else{  
                b[0]+=(ma-a[i])/2;  
                b[1]+=((ma-a[i])/2+(ma-a[i])%2);  
            }  
        }  
        if(b[0]==b[1])cout<<"YES"<<endl;  
        else{  
            if(b[0]-b[1]==1&&n%2)cout<<"YES"<<endl;  
            else cout<<"NO"<<endl;  
        }  
    }  
      
} 

第二种

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值