邻面合并(merging)

题目描述

给定一个N×MN×M的网格,每个格子上写有0或1。现在用一些长方形覆盖其中写有1的格子,长方形的每条边都要与坐标轴平行。要求:每个写着1的格子都要被覆盖,长方形不可以重叠(重复绘制也多少会增加性能开销),也不能覆盖到任何一个写着0的格子(不然绘制结果就不正确了)。请问最少需要多少长方形?

输入

输入文件第一行两个正整数N,MN,M,表示网格大小为NN行MM列。

接下来的NN行,每行MM个正整数AijAij(保证均为0或1),其中第ii行jj列的正整数表示网格ii行jj列里填的数。

输出

输出文件包含一行一个正整数,表示最少需要的长方形数量。

样例输入

<span style="color:#333333"><span style="color:#333333">4 4
1 1 1 0
1 1 1 1
0 0 1 1
0 0 1 1</span></span>

样例输出

<span style="color:#333333"><span style="color:#333333">3</span></span>

提示

样例解释

一种行的覆盖方案(粗线表示分割线):

数据范围

对于30% 的数据:N,M≤5N,M≤5。

对于100% 的数据:N≤100,M≤8N≤100,M≤8。

来源


solution

状压dp,我想不到

令f[i][S]表示前i行,状态为S

状态为1表示是一个新的开始

比如状态10010

表示划成了2个矩形,开头在1 4

转移时枚举是否能接起来

#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define inf 1e9
using namespace std;
int n,m,s[105][10],f[102][1<<9];
int num=0;
bool pd(int k,int S){
for(int i=0;i<m;i++){
int t=(1<<i);
if((S&t)&&s[k][i]==0)return 0;
}
int fl=0;
for(int i=0;i<m;i++){
int t=(1<<i);
if(S&t)fl=1;
if(s[k][i]==1&&!fl)return 0;
if(s[k][i]==0)fl=0;
}
return 1;
}
int cost(int k,int S,int T){
int sum=0;
for(int i=0;i<m;i++){
int t=(1<<i);
if(S&t)sum++;
} for(int i=0;i<m;i++){
int t=(1<<i);
if((S&t)&&(T&t)){
int ed=i;for(;s[k][ed]&&ed<=m;ed++)if(ed!=i&&(S&(1<<ed)))break;
//cout<<"ed "<<ed<<' '<<i<<endl;
bool fl=0;
for(int j=i+1;j<ed;j++)if(T&(1<<j)){fl=1;break;}
for(int j=i+1;j<ed;j++)if(!s[k-1][j]){fl=1;break;}
if(!fl&&((T&(1<<ed))||!s[k-1][ed]))sum--;
}
}
return sum;
}
int main()
{
cin>>n>>m;
for(int i=1;i<=n;i++){
for(int j=0;j<m;j++){
scanf("%d",&s[i][j]);
} }
for(int i=0;i<=n;i++)
for(int j=0;j<(1<<m);j++)f[i][j]=inf;
f[0][0]=0;
for(int i=1;i<=n;i++){
for(int S=0;S<(1<<m);S++){
if(pd(i,S)){
for(int T=0;T<(1<<m);T++){
if(f[i-1][T]!=inf){
//cout<<i<<' '<<S<<' '<<T<<endl; f[i][S]=min(f[i][S],f[i-1][T]+cost(i,S,T));
//cout<<cost(i,S,T)<<endl;
//system("pause");
}
}
}
}
}
int ans=inf;
for(int S=0;S<(1<<m);S++)ans=min(ans,f[n][S]);
cout<<ans<<endl;
return 0;
}

邻面合并(merging)的更多相关文章

  1. [CSP-S模拟测试]:邻面合并(状压DP)

    题目背景 $NEWorld$作为一个$3D$游戏,对渲染(图形绘制)的效率要求极高.当玩家扩大视野范围时,可见的方块面数量将会迅速增多,以至于大量的顶点处理很快就成为了图形管线中的瓶颈.乔猫想了想,决 ...

  2. [HNOI2009]梦幻布丁 算法技巧之邻接链

    题目描述 N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1,2,2,1的四个布丁一共有3段颜色. 输入输出格式 输入格式: 第 ...

  3. 理解 Git 的基本概念 ( Merging Collaborating Rebasing)

    合并 Merging 在分支上开发新功能后,如何把新功能加入到主分支,让其它人得到你的修改呢?你需要使用命令 git merge 或 git pull. 这两个命令的语法如下: git merge [ ...

  4. NOIP模拟80

    学考+OJ改名祭 T1 邻面合并 解题思路 状压 DP ...(于是贪心竟然有 60pts 的高分?? code) 状态设计的就非常妙了,如果状态是 1 就表示是一个分割点也就是一个矩形的右边界. 那 ...

  5. Noip模拟80 2021.10.18

    预计得分:5 实际得分:140?????????????? T1 邻面合并 我考场上没切掉的大水题....(证明我旁边的cty切掉了,并觉得很水) 然而贪心拿了六十,离谱,成功做到上一篇博客说的有勇气 ...

  6. elasticseach multi-field的实际用途

    下面是multi-field的介绍: multi_field 多域类型允许你对同一个值以映射的方式定义成多个基本类型 core_types . 这个非常有用,比如,如果你定义一个 string 类型的 ...

  7. TortoiseSVN中图标的含义

    今天在使用svn时发现有好多不认识了,所以查了下svn帮助手册.借此总结了下 svn 中图标的含义 一个新检出的工作复本使用绿色的勾做重载.表示Subversion状态 正常. 在开始编辑一个文件后, ...

  8. C语言动态内存管理

    1-概述 动态存储管理的基本问题是:系统如何按请求分配内存,如何回收内存再利用.提出请求的用户可能是系统的一个作业,也可能是程序中的一个变量. 空闲块 未曾分配的地址连续的内存区称为“空闲块”. 占用 ...

  9. Git让你从入门到精通,看这一篇就够了!

    简介 Git 是什么? Git 是一个开源的分布式版本控制系统. 什么是版本控制? 版本控制是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统. 什么是分布式版本控制系统? 介绍分布 ...

随机推荐

  1. C# 创建和初始化集合对象

    一. 引言 C# 3.0中新的对象初始化器是一种简单的语法特征-借助于这种特征,对象的构建和初始化变得非常简单.假定你有一个类Student,它看起来有如下样子: public class Stude ...

  2. linux apache 不解析php文件显示源码

    首先检查是否安装PHP 没有的话就先安装 如果安装过 在/etc/httpd/conf/httpd.conf文件中 在<IfModule mime_module>里面 AddType ap ...

  3. Sql优化器究竟帮你做了哪些工作?

    关系型数据库的一大优势之一,用户无需关心数据的访问方式,因为这些优化器都帮我们处理好了,但sql查询优化的时候,我不得不要对此进行关注,因为这牵扯到查询性能问题. 有经验的程序员都会对一些sql优化了 ...

  4. 更改 Linux 语言为中文

    查看当前系统语言环境:    echo $LANG 查看安了哪些中文语言包    locale -a |grep "zh_CN" 没有输出,说明没有安装,输入下面的命令安装     ...

  5. node 中的redis使用

    1.创建sql.config.js 配置文件 : var redis_db = { ", "URL":"127.0.0.1", "OPTIO ...

  6. 【Python学习之七】面向对象高级编程——使用@property

    参考来自廖雪峰Python教程:https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/ ...

  7. 让你提高效率的 Linux 技巧

    想要在 Linux 命令行工作中提高效率,你需要使用一些技巧. 巧妙的 Linux 命令行技巧能让你节省时间.避免出错,还能让你记住和复用各种复杂的命令,专注在需要做的事情本身,而不是你要怎么做.以下 ...

  8. Python学习笔记:装饰器

    Python 装饰器的基本概念和应用 代码编写要遵循开放封闭原则,虽然在这个原则是用的面向对象开发,但是也适用于函数式编程,简单来说,它规定已经实现的功能代码不允许被修改,但可以被扩展,即: 封闭:已 ...

  9. Find a path HDU - 5492 (dp)

    Find a path Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  10. HDU 6153 KMP

    最终刷KMP目标就是为了挑战这道题!现在成功了恩... 首先,题目大意是:给出一个字符串str1,之后给出另一个字符串str2,问,str2的后缀在str1匹配的次数*后缀当前长度是多少 首先考虑正统 ...