棋盘分割 (POJ - 1191,二维区间 DP)

2024-03-30 13:48

本文主要是介绍棋盘分割 (POJ - 1191,二维区间 DP),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一.题目链接:

POJ-1191

二.题目大意:

中文题...

三.分析:

卧艹,这尼玛也太暴力了!

大佬讲解1   大佬讲解2

一开始没想明白状态表示,是我太菜.

dp[i][x1][y1][x2][y2] 表示 将区域 (x1, y1, x2, y2) 切 i 次后的子矩形平方和最小值.

搞清状态表示后,状态转移就很好写啦.

详见代码(没错,我就是懒得讲)

四.代码实现:

#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;const int M = 10;
const int inf = 0x3f3f3f3f;int sum[M][M];
int dp[M + 5][M][M][M][M];int cost(int x1, int y1, int x2, int y2)
{int s = sum[x2][y2] - sum[x2][y1 - 1] - sum[x1 - 1][y2] + sum[x1 - 1][y1 - 1];return s * s;
}int main()
{int n;scanf("%d", &n);for(int i = 1; i <= 8; ++i){for(int j = 1; j <= 8; ++j)scanf("%d", &sum[i][j]);}for(int i = 1; i <= 8; ++i){for(int j = 1; j <= 8; ++j)sum[i][j] += sum[i][j - 1] + sum[i - 1][j] - sum[i - 1][j - 1];}memset(dp, inf, sizeof(dp));for(int x1 = 1; x1 <= 8; ++x1){for(int y1 = 1; y1 <= 8; ++y1){for(int x2 = x1; x2 <= 8; ++x2){for(int y2 = y1; y2 <= 8; ++y2){dp[0][x1][y1][x2][y2] = cost(x1, y1, x2, y2);}}}}for(int i = 1; i <= n - 1; ++i){for(int x1 = 1; x1 <= 8; ++x1){for(int y1 = 1; y1 <= 8; ++y1){for(int x2 = x1; x2 <= 8; ++x2){for(int y2 = y1; y2 <= 8; ++y2){for(int x = x1; x < x2; ++x)dp[i][x1][y1][x2][y2]= min(dp[i][x1][y1][x2][y2],min(dp[i - 1][x1][y1][x][y2] + dp[0][x + 1][y1][x2][y2],dp[i - 1][x + 1][y1][x2][y2] + dp[0][x1][y1][x][y2]));for(int y = y1; y < y2; ++y)dp[i][x1][y1][x2][y2]= min(dp[i][x1][y1][x2][y2],min(dp[i - 1][x1][y1][x2][y] + dp[0][x1][y + 1][x2][y2],dp[i - 1][x1][y + 1][x2][y2] + dp[0][x1][y1][x2][y]));}}}}}printf("%.3f\n", sqrt(1.0 * dp[n - 1][1][1][8][8] / n - 1.0 * sum[8][8] * sum[8][8] / n / n));return 0;
}

 

这篇关于棋盘分割 (POJ - 1191,二维区间 DP)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/861651

相关文章

上海邀请赛 A题目 HDU 5236(dp)

先求出没有ctrl+s的时候构造长度为i的期望f[i] 。然后枚举保存的次数,求出最小即可。 #include<cstdio>#include<cstdio>#include<cmath>#include<queue>#include<stack>#include<string>#include<cstring>#include<iostream>#include<map>

poj 3882(Stammering Aliens) 后缀数组 或者 hash

后缀数组:  构建后缀数组,注意要在字符串莫末尾加上一个没出现过的字符。然后可以2分或者直接扫描,直接扫描需要用单调队列来维护 VIEW CODE #include<cstdio>#include<algorithm>#include<iostream>#include<cmath>#include<queue>#include<stack>#include<string

poj 3294(Life Forms) 2分+ 后缀数组

我曾用字符串hash写,但是超时了。只能用后最数组了。大致思路:用不同的符号吧字符串连接起来,构建后缀数组,然后2分答案,依次扫描后缀数组,看是否瞒住条件。 VIEW CODE #include<cstdio>#include<vector>#include<cmath>#include<algorithm>#include<cstring>#include<cassert>#

poj 2391 Ombrophobic Bovines (网络流)

这是一道很经典的网络流的题目。首先我们考虑假如我们的时间为无穷大。我们吧每个点拆成2个点 i和i' .。虚拟源点s和汇点t。对于每个点建边(s,i, a[i])  (i‘,t,ib[i]) 。 其中a[i]为给点有多少牛,b[i]为容量。i和j连通 建边 (i,j',inf);如果最大流==所有牛的个数,就可能装下所有的牛。那么现在我们考虑时间。假设最大时间为T.那么如果i到j的的最短时间>T

poj 1330 LCA 最近公共祖先

水题目。直接上代码了。 VIEW CODE #include<cstdio>#include<algorithm>#include<iostream>#include<cmath>#include<queue>#include<stack>#include<string>#include<cstring>#include<map>#include<vector>#

poj 3160 Father Christmas flymouse 强连通+dp

首先我们可以确定的是,对于val值小于0的节点都变成0.   假设一个集合内2个房间都能任意到达,那么我就可以吧集合内的所有点的价值都取到,并且可以达到任一点。实际上集合内的每个点是相同的,这样的集合就是一个强连通分量。 那么我们就可以用tarjin算法进行强连通缩点, 最后形成一个dag的图。在dag的图上面进行dp。可以先用拓扑排序后dp。或者建反响边记忆化搜索 。 VIEW

秋招突击——6/22——复习{区间DP——加分二叉树,背包问题——买书}——新作{移除元素、实现strStr()}

文章目录 引言复习区间DP——加分二叉树个人实现 背包问题——买书个人实现参考实现 新作移除元素个人实现参考思路 找出字符串中第一个匹配项的下标个人实现参考实现 总结 引言 今天做了一个噩梦,然后流了一身汗,然后没起来,九点多才起床背书。十点钟才开始把昨天那道题题目过一遍,然后十一点才开始复习题目,为了不耽误下午的时间,所以这里的就单纯做已经做过的题目,主打一个有量,不在学

leetcode刷题(95)——416. 分割等和子集

给定一个只包含正整数的非空数组。是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。 注意: 每个数组中的元素不会超过 100 数组的大小不会超过 200 示例 1: 输入: [1, 5, 11, 5]输出: true解释: 数组可以分割成 [1, 5, 5] 和 [11]. 示例 2: 输入: [1, 2, 3, 5]输出: false解释: 数组不能分割成两个元素和相等的子

动态规划DP--斐波那契数、爬楼梯、使用最小花费爬楼梯等示例代码

动态规划DP 文章目录 动态规划DP509. 斐波那契数70. 爬楼梯746. 使用最小花费爬楼梯62. 不同路径63. 不同路径II343.整数拆分 509. 斐波那契数 509. 斐波那契数 斐波那契数 (通常用 F(n) 表示)形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始,后面的每一项数字都是前面两项数字的和。也就是: F(0) = 0,F(1) =

Java中的二维数组和c语言中的二维数组的区别

我觉得,JAVA的多维数组其实是数组包数组,即他们下一个数组是独立的,可以独立分配内存大小,跟C语言的数组不一样,C语言的数组无论维数是多少,他们每一维的内存大小都一样。 打个比方: JAVA的三维数组 某公司有m个工厂,这个是第一维; 每个工厂有n个仓库,这个是第二维; 每个仓库有o件库存,这是第三维; 每个工厂的仓库数量都不同,每个仓库的库存数量又都不同。 通过