棋盘分割
Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 16263   Accepted: 5812

Description

将一个8*8的棋盘进行如下分割:将原棋盘割下一块矩形棋盘并使剩下部分也是矩形,再将剩下的部分继续如此分割,这样割了(n-1)次后,连同最后剩下的矩形棋盘共有n块矩形棋盘。(每次切割都只能沿着棋盘格子的边进行) 

原棋盘上每一格有一个分值,一块矩形棋盘的总分为其所含各格分值之和。现在需要把棋盘按上述规则分割成n块矩形棋盘,并使各矩形棋盘总分的均方差最小。 
均方差,其中平均值,xi为第i块矩形棋盘的总分。 
请编程对给出的棋盘及n,求出O'的最小值。 

Input

第1行为一个整数n(1 < n < 15)。 
第2行至第9行每行为8个小于100的非负整数,表示棋盘上相应格子的分值。每行相邻两数之间用一个空格分隔。 

Output

仅一个数,为O'(四舍五入精确到小数点后三位)。

Sample Input

3
1 1 1 1 1 1 1 3
1 1 1 1 1 1 1 1
1 1 1 1 1 1 1 1
1 1 1 1 1 1 1 1
1 1 1 1 1 1 1 1
1 1 1 1 1 1 1 1
1 1 1 1 1 1 1 0
1 1 1 1 1 1 0 3

Sample Output

1.633

Source

题意:

一个8*8的棋盘,进行分割。每次将一个矩形分割成两个。一个矩形的值是里面所有格子值之和。现在想切出n个矩形,希望求得最小的均方差。

思路:

我们先把均方差的公式进行化简

平均值是一个定值,因为每个矩形都是格子值之和。于是我们发现结果之和Xi平方有关。Xi平方越小越好。

对于每一个格子,都可以用两个坐标(i, j)和(x,y)表示,他们分别是矩形的左上角和右下角。

一个矩形有两种切割方法,横着切,竖着切,假设在k处切

横着切时,矩形被分成了(i, j)(k, y) 和 (k + 1, j)(x,y)

竖着切时,矩形被分成了(i,j)(x,k) 和(i, k+1)(x,y)

但是这样还没办法进行状态转移,因为矩形的先后顺序不知道。所以我们可以再引入一维变量,表示各自的顺序。

于是在切割i次时,得到两个矩形,其中一个应该是i+1次切割的矩形。可以得到状态转移方程。

因为是i推i+1,所以用记忆化搜索写起来可能方便一点

 //#include <bits/stdc++.h>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<stdio.h>
#include<cstring>
#include<vector>
#include<map>
#include<set> #define inf 0x3f3f3f3f
using namespace std;
typedef long long LL; int n;
int board[][], hang[][], dp[][][][][]; /*int getval(int i, int j, int x, int y)
{
int res = 0;
for(int k = i; k <= x; k++){
res += hang[i][y] - hang[i][j - 1];
}
return res * res;
}*/ int getval(int i, int j, int x, int y)
{
int res = ;
for(int a = i; a <= x; a++){
for(int b = j; b <= y; b++){
res += board[a][b];
}
}
return res * res;
} int DP(int k, int i, int j, int x, int y)
{
int ans = ;
if(dp[k][i][j][x][y] >= )return dp[k][i][j][x][y];
if(k == n - ){
return getval(i, j, x, y);
}
dp[k][i][j][x][y] = inf;
for(int tmp = i; tmp < x; tmp++){
ans = min(DP(k + , i, j, tmp, y) + getval(tmp + , j, x, y), DP(k + , tmp + , j, x, y) + getval(i, j, tmp, y));
dp[k][i][j][x][y] = min(ans, dp[k][i][j][x][y]);
}
for(int tmp = j; tmp < y; tmp++){
ans = min(DP(k + , i, j, x, tmp) + getval(i, tmp + , x, y), DP(k + , i, tmp + , x, y) + getval(i, j, x, tmp));
dp[k][i][j][x][y] = min(ans, dp[k][i][j][x][y]);
}
return dp[k][i][j][x][y];
} int main(){ while(scanf("%d", &n) != EOF){
double sum = ;
memset(hang, , sizeof(hang));
for(int i = ; i <= ; i++){
for(int j = ; j <= ; j++){
scanf("%d", &board[i][j]);
sum += board[i][j] * 1.0;
hang[i][j] = hang[i][j - ] + board[i][j];
}
}
sum /= 1.0 * n;
memset(dp, -, sizeof(dp));
int ans = DP(, , , , );
//cout<<ans<<endl;
//cout<<dp[n - 1][1][1][8][8]<<endl;
printf("%.3f\n",sqrt((dp[][][][][])*1.0/n-sum*sum));
}
return ;
}

poj1191 棋盘分割【区间DP】【记忆化搜索】的更多相关文章

  1. HDU 2517 / POJ 1191 棋盘分割 区间DP / 记忆化搜索

    题目链接: 黑书 P116 HDU 2157 棋盘分割 POJ 1191 棋盘分割 分析:  枚举所有可能的切割方法. 但如果用递归的方法要加上记忆搜索, 不能会超时... 代码: #include& ...

  2. (区间dp + 记忆化搜索)Treats for the Cows (POJ 3186)

    http://poj.org/problem?id=3186   Description FJ has purchased N (1 <= N <= 2000) yummy treats ...

  3. POJ 1191 棋盘分割 【DFS记忆化搜索经典】

    题目传送门:http://poj.org/problem?id=1191 棋盘分割 Time Limit: 1000MS   Memory Limit: 10000K Total Submission ...

  4. UVA 10003 Cutting Sticks 区间DP+记忆化搜索

    UVA 10003 Cutting Sticks+区间DP 纵有疾风起 题目大意 有一个长为L的木棍,木棍中间有n个切点.每次切割的费用为当前木棍的长度.求切割木棍的最小费用 输入输出 第一行是木棍的 ...

  5. uva 10891 区间dp+记忆化搜索

    https://vjudge.net/problem/UVA-10891 给定一个序列x,A和B依次取数,规则是每次只能从头或者尾部取走若干个数,A和B采取的策略使得自己取出的数尽量和最大,A是先手, ...

  6. poj1191棋盘分割——区间DP

    题目:http://poj.org/problem?id=1191 分析题意,可知每次要沿棋盘中的一条线把一块一分为二,取其中一块继续分割: σ最小经分析可知即为每块的xi和的平方最小: 故用区间DP ...

  7. loj 1031(区间dp+记忆化搜索)

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1031 思路:dp[i][j]表示从区间i-j中能取得的最大值,然后就是枚举分割点了. ...

  8. BZOJ1055[HAOI2008]玩具取名 【区间dp + 记忆化搜索】

    题目 某人有一套玩具,并想法给玩具命名.首先他选择WING四个字母中的任意一个字母作为玩具的基本名字.然后 他会根据自己的喜好,将名字中任意一个字母用“WING”中任意两个字母代替,使得自己的名字能够 ...

  9. hdu 4597 Play Game(区间dp,记忆化搜索)

    Problem Description Alice and Bob are playing a game. There are two piles of cards. There are N card ...

  10. poj 1088 滑雪(区间dp+记忆化搜索)

    题目链接:http://poj.org/problem?id=1088 思路分析: 1>状态定义:状态dp[i][j]表示在位置map[i][j]可以滑雪的最长区域长度: 2>状态转移方程 ...

随机推荐

  1. 在windows下编译ffmpeg

    编译ffmpeg,我在网上找了很多相关的方法,但最后都没编译成功. 所以下面就记录下自己的编译方法吧,留着以后编译的时候做参考. 1.首先,下载编译工具MinGW+Msys,搭建编译环境.工具下载地址 ...

  2. css -- 通俗理解inline、block、inline-block

    display:inline; 内联元素,简单来说就是在同一行显示. display:block; 块级元素,简单来说就是就是有换行,会换到第二行. display:inline-block; 就是在 ...

  3. chrome 版本升级到56,报错Your connection is not private NET::ERR_CERT_WEAK_SIGNATURE_ALGORITHM

    ubuntu14.04 解决方法: $ sudo apt-get install libnss3-1d ref: http://stackoverflow.com/questions/42085055 ...

  4. 启动vsftpd失败

    启动vsftpd失败 在使用centos时, 要用ftp上传文件, 但是一到脚本的ftp命令就会出错: rpm -Uvh http://mirror.centos.org/centos/6/os/i3 ...

  5. oracle的varchar2和clob类型在hibernate中使用

    1.在oracle中 varchar2的最大长度为4000 bytes,即varchar2(4000),最多能储存2000个汉子或4000位的数字字母.当储存值超过时可以使用clob(Characte ...

  6. php错误:You don't have permission to access / on this server.

    以前php环境崩溃了,重新装了个,打开第一个文件就出现: You don't have permission to access / on this server.错误,让我情以何堪啊,居然说我此台服 ...

  7. win7 安装 VMware 出错解决办法

    win7旗舰版安装VMware,安装过程中发生了如下错误.系统提示:“You may not install this product in the root directory of any dri ...

  8. 关于Java 枚举类型的自定义属性

    package com.cpic.test;/** * 关于枚举类型自定义属性 * */public enum Provious { ANHUI("皖", 1),BAIJING(& ...

  9. Windows7 64bits下安装TensorFlow CPU版本(图文详解)

    不多说,直接上干货! Installing TensorFlow on Windows的官网 https://www.tensorflow.org/install/install_windows 首先 ...

  10. laravel 集合接口

    只记下几个常用的,其他的看这里:http://laravelacademy.org/post/6293.html 1)什么是集合? 就是laravel查询构建器查询返回的数据结果(get first ...