搜索与DP:SLIKAR
Problem: SLIKAR
【题目描述】
Josip 是个奇怪的画家,他想画一幅由 N*N 个点组成的图, N 是一个 2 的乘方
数(1, 2, 4, 8, 16 等.)。每个点要么是黑色的,要么是白色的。 Josip 画画有一个习惯。
他用下列递归的方式画画:
1.如果图像只包含一个点,他可以以任意色画(黑或白)。
2.否则,他把图分成四个大小相等的正方形部分:
3.从四个中选出一个,用白色的填充。
4.从剩余的三个中选一个,用黑色的填充。
5.把剩余的二部分重新按相同的三步进行填充。
不久以后,他发现,用他的方法并不能画出所有的画。请你编一个程序帮他设计一
个画图方案,使得用他的方法画出的画与想得到的画之间的差距最少(不同点的个
数最少)。
【输入说明】
第一行为一个整数 N (1 ≤ N ≤ 512),表示 Josip 画的图的尽寸。 N 是 2 的乘方数。
后面有 N 行,每行由 N 个 0 或 1 组成,表示画中的黑白点。
【输出说明】
输出第一行为最少可能的不同点数。
后面 N 行输出 Josip 可以画出的与目标图最接近的图。
注,第二部分的图可能不唯一。
【约定】
有 50%的点, N 最多是 8。
【样例】
input
4
1111
1111
1111
1111
output
6
0011
0011
0111
1101
这题其实暴搜就可以了,这里我建了一棵四叉树,用来处理二维的区间问题
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int N,map[][],tot[][];
char s[][];
int work[][]={
{},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
{,,,,},
};
struct node{
int L,R,U,D,K,sum;
}tr[];
int Q(int U,int D,int L,int R)
{
return tot[D][R]-tot[U-][R]-tot[D][L-]+tot[U-][L-];
}
int S(int U,int D,int L,int R)
{
return (D-U+)*(R-L+);
}
void Build(int Node,int U,int D,int L,int R)
{
tr[Node].L=L;tr[Node].R=R;
tr[Node].U=U;tr[Node].D=D;
if(L==R)return;
Build(*(Node-)+,U,(U+D)/,L,(L+R)/);
Build(*(Node-)+,U,(U+D)/,(L+R)/+,R);
Build(*(Node-)+,(U+D)/+,D,L,(L+R)/);
Build(*(Node-)+,(U+D)/+,D,(L+R)/+,R);
} void Solve(int Node)
{
if(tr[Node].L==tr[Node].R){
tr[Node].sum=;
return;
}
int son[];
son[]=*(Node-)+;son[]=*(Node-)+;
son[]=*(Node-)+;son[]=*(Node-)+; Solve(son[]);Solve(son[]);
Solve(son[]);Solve(son[]); int minn=,minp;
int ss=S(tr[Node].U,tr[Node].D,tr[Node].L,tr[Node].R)/;
for(int i=;i<=;i++)
{
int ret;
int a=son[work[i][]],b=son[work[i][]];
int c=son[work[i][]],d=son[work[i][]];
ret=tr[a].sum+tr[b].sum;
ret+=Q(tr[c].U,tr[c].D,tr[c].L,tr[c].R);
ret+=ss-Q(tr[d].U,tr[d].D,tr[d].L,tr[d].R);
if(ret<minn){
minn=ret;
minp=i;
}
}
int a=son[work[minp][]],b=son[work[minp][]];
int c=son[work[minp][]],d=son[work[minp][]];
tr[a].K=tr[b].K=;
tr[c].K=;tr[d].K=;
tr[Node].sum=minn;
return;
} void Paint(int Node)
{
int son[];
son[]=*(Node-)+;son[]=*(Node-)+;
son[]=*(Node-)+;son[]=*(Node-)+; if(tr[Node].K==){
if(tr[Node].L==tr[Node].R)
return;
Paint(son[]);Paint(son[]);
Paint(son[]);Paint(son[]);
}
else{
for(int i=tr[Node].U;i<=tr[Node].D;i++)
for(int j=tr[Node].L;j<=tr[Node].R;j++)
map[i][j]=tr[Node].K;
}
}
int main()
{
freopen("slikar.in","r",stdin);
freopen("slikar.out","w",stdout);
scanf("%d",&N);
for(int i=;i<=N;i++)scanf("%s",s[i]+);
for(int i=;i<=N;i++)
for(int j=;j<=N;j++)
{
map[i][j]=s[i][j]-'';
map[i][j]+=map[i][j-];
} for(int i=;i<=N;i++)
for(int j=;j<=N;j++)
tot[i][j]=tot[i-][j]+map[i][j]; Build(,,N,,N); Solve(); for(int i=;i<=N;i++)
for(int j=;j<=N;j++)
map[i][j]=s[i][j]-''; tr[].K=;
Paint(); printf("%d\n",tr[].sum);
for(int i=;i<=N;i++){
for(int j=;j<=N;j++)
printf("%d",map[i][j]);
printf("\n");
}
return ;
}
搜索与DP:SLIKAR的更多相关文章
- 记忆化搜索(DFS+DP) URAL 1223 Chernobyl’ Eagle on a Roof
题目传送门 /* 记忆化搜索(DFS+DP):dp[x][y] 表示x个蛋,在y楼扔后所需要的实验次数 ans = min (ans, max (dp[x][y-i], dp[x-1][i-1]) + ...
- 记忆化搜索(DFS+DP) URAL 1501 Sense of Beauty
题目传送门 /* 题意:给了两堆牌,每次从首部取出一张牌,按颜色分配到两个新堆,分配过程两新堆的总数差不大于1 记忆化搜索(DFS+DP):我们思考如果我们将连续的两个操作看成一个集体操作,那么这个操 ...
- 搜索+简单dp
前言:即使是简单的递归,在复杂度过高时也可以使用简单的dp. 一般有两种情况,一是利用dp思想求最优子结构进行搜索剪枝,二是利用搜索进行dp数组的填充. 例题一.hdu1978 题目大意:这是一个简单 ...
- UVALive 4864 Bit Counting --记忆化搜索 / 数位DP?
题目链接: 题目链接 题意:如果一个数二进制n有k位1,那么f1[n] = k,如果k有s位二进制1,那么f2[n] = f1[k] = s. 如此往复,直到fx[n] = 1,此时的x就是n的”K ...
- hdu3555 Bomb (记忆化搜索 数位DP)
http://acm.hdu.edu.cn/showproblem.php?pid=3555 Bomb Time Limit: 2000/1000 MS (Java/Others) Memory ...
- HDU 2476 String painter(记忆化搜索, DP)
题目大意: 给你两个串,有一个操作! 操作时可以把某个区间(L,R) 之间的所有字符变成同一个字符.现在给你两个串A,B要求最少的步骤把A串变成B串. 题目分析: 区间DP, 假如我们直接想把A变成B ...
- hdu_3562_B-number(记忆化搜索|数位DP)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=3652 题意:给你一个n,为比n小的能整除13并数字中有13的数有多少个 题解:记忆化搜索:记dp[i] ...
- CF1038E Maximum Matching 搜索/区间DP
题目传送门:http://codeforces.com/problemset/problem/1038/E 题意:给出$N$个方块,每个方块有左右两种颜色$a,b$(可以翻转使左右两种颜色交换)和一个 ...
- POJ-1088 滑雪 (记忆化搜索,dp)
滑雪 Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 86318 Accepted: 32289 Description Mich ...
随机推荐
- Weex 初始
1.一旦数据和模板绑定,数据的变化会立即体现在前台的变化 <template> <container> <text style="font-size: {{si ...
- object标签参考(转载)
<object> 元素可支持多种不同的媒介类型,比如: 图片 音频 视频 Other 对象 显示图片 你可以显示一幅图片: <object height="100%&quo ...
- android调用系统图片浏览器裁切后出现黑边
是这样的:我使用系统的图片浏览器,然后让它自动跳到图片裁切界面,当我们定义了返回的图片大小过大,而我们实际的图片像素达不到时,系统为我们自动地填充了不够的像素成黑色,那么我们怎么样来解决这个问题呢?不 ...
- 段落排版--中文字间距、字母间距(letter-spacing, word-spacing)
中文字间隔.字母间隔设置: 如果想在网页排版中设置文字间隔或者字母间隔就可以使用 letter-spacing 来实现,如下面代码: h1{ letter-spacing:50px; } ... ...
- Jmeter软件测试2--http接口测试
上次利用Jmeter进行了webservice接口的测试,本次利用Jmeter进行http接口的测试 1.新建线程组 2.新建配置文件 3.新建http请求 4.配置动态请求 4.查看测试结果
- java基础之抽象类与接口的区别
在学习java的过程中,或者是在找工作笔试或面试的时候,如果你是java岗位,那么抽象类与接口的区别无疑是一个大热点,是各大公司都想要考的一个小知识点,下面是我为了9月份秋招自己总结的,若有不对的地方 ...
- 浅谈KMP算法及其next[]数组
KMP算法是众多优秀的模式串匹配算法中较早诞生的一个,也是相对最为人所知的一个. 算法实现简单,运行效率高,时间复杂度为O(n+m)(n和m分别为目标串和模式串的长度) 当字符串长度和字符集大小的比值 ...
- yum安装ftp服务器
1.安装vsftp,本文采用yum安装: #yum install vsftpd 2.安装后运行: # service vsftpd restart Shutting downvsftpd: ...
- overflow第一次觉得你有点可恶
今天用css做下拉菜单,因为不需要做手机自适应,再手机里看起来工整一点就行,可是列表中最后一个li的宽度撑开了父div,导致看起来很糟糕,所以给父元素加overflow:hidden:但是下拉列表也被 ...
- ICE学习第三步-----Slice语言
ICE:Slice语言(一)-编译 Introduce简介 Slice(Specification language for ice)是分离对象和对象的实现的基础的抽象机制.Slice在客户端和服务器 ...