POJ 1191棋盘分割问题
2024-08-26 19:59:39
题目大意,将一个棋盘分割成k-1个矩形,每个矩形都对应一个权值,让所有的权值最小求分法
很像区间DP,但是也不能说就是
我们只要想好了一个怎么变成两个,剩下的就好了,但是怎么变,就是变化的必要条件是什么
k——分割的个数肯定是必须的,而表示一个矩形,至少要知道两个点,所以x1,y1,x2,y2也是必须的,So,五维的DP,以前想都不敢想啊
dp[k][x1][y1][x2][y2]
先不来说他的值如何计算,先来看看如何分割
根据区间DP的思想
dp[k][x1][y1][x2][y2]
如果横向分割就会有一个状态 dp[k-1][x1][y1][x2][t] + dp[0][x1][t+1][x2][y2]
dp[0][x1][y1][x2][t] + dp[k-1][x1][t+1][x2][y2]
相对应竖向呢就会有 dp[k-1][x1][y1][t][y2] + dp[0][t+1][y1][x2][y2]
dp[0][x1][y1][t][y2] + dp[k-1][t+1][y1][x2][y2]
这就是状态转移的方程
最后就是权值的问题了
对方差公式进行化解,得到σ^2=1/n∑xi^2 - x^2
可知,要使方差最小,只需使∑xi^2最小即可,即各块分值平方和最小。平均值 x是个固定的数,跟分割的方式没有关系,
#include <iostream>
#include <cstdio>
#include <cmath>
#include <string.h>
#include <iomanip>
using namespace std;
int data[9][9];
int sum[9][9];
double dp[14][9][9][9][9]; double get_count(int x1,int y1,int x2,int y2)
{
double ans = double(sum[x2][y2] - sum[x1-1][y2]-sum[x2][y1-1] + sum[x1-1][y1-1]); return ans * ans;
}
int main()
{
int n,total = 0;
scanf("%d",&n);
for(int i = 1;i <= 8;i++)
{
for(int j = 1;j <= 8;j++)
{
cin>>data[i][j];
//sum[i][j]表示棋盘(1,1)到(i,j)区域的累计分值
sum[i][j] = sum[i][j-1] + sum[i-1][j] - sum[i-1][j-1] + data[i][j];
//total表示整个棋盘的分值之和
total += data[i][j];
}
} 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] = get_count(x1,y1,x2,y2);
}
}
}
}
for(int k = 1;k < n;k++)
{
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++)
{
int t;
dp[k][x1][y1][x2][y2] = (double)(1 << 30);
for(t = x1;t < x2;t++)
{
dp[k][x1][y1][x2][y2] = min(dp[k][x1][y1][x2][y2],dp[0][x1][y1][t][y2] + dp[k-1][t+1][y1][x2][y2]);
dp[k][x1][y1][x2][y2] = min(dp[k][x1][y1][x2][y2],dp[k-1][x1][y1][t][y2] + dp[0][t+1][y1][x2][y2]);
} for(t = y1;t < y2;t++)
{
dp[k][x1][y1][x2][y2] = min(dp[k][x1][y1][x2][y2],dp[0][x1][y1][x2][t] + dp[k-1][x1][t+1][x2][y2]);
dp[k][x1][y1][x2][y2] = min(dp[k][x1][y1][x2][y2],dp[k-1][x1][y1][x2][t] + dp[0][x1][t+1][x2][y2]);
}
}
}
}
}
}
double ans = dp[n-1][1][1][8][8] * 1.0 / n - ((double)total*1.0/n)*((double)total*1.0/n);
//printf("%d %.3lf\n",total,dp[n-1][1][1][8][8]);
printf("%.3lf\n",sqrt(ans));
return 0;
}
最新文章
- Windows Server 2008 任务计划无法自动运行的解决办法
- MyEclipse 8.5 优化实例
- selenium 一个简单的流程
- head First HTML与CSS读书笔记
- Linux系统时间和硬件时间设置
- php写一个简洁的登录页面
- docker 简单介绍及基础命令运用
- 【QT】二进制读取图像文件测试
- 跨平台设置NODE_ENV(兼容win和linux)
- 如何用 async 控制流程
- event.currentTarget和event.target的区别
- 理解%r和%s的区别
- go语言之进阶篇Read的使用
- 【Oracle】详解Oracle中的序列
- 【appium】根据UIAutomator定位元素
- ubuntu初次设置root密码
- Jquery 简明介绍
- How to Configure Eclipse for Python --- 在eclipse中如何配置pydev
- 进阶篇:4.1)DFA设计指南:简化产品设计(kiss原则)
- ubuntn 安装软件
热门文章
- 为什么CNN能自动提取图像特征
- [中英对照]Introduction to DPDK: Architecture and Principles | DPDK概论: 体系结构与实现原理
- Linux 终端设备
- tp5查看版本
- Java中 i++ 是线程安全的么?为什么?
- Spring 注解原理(三)@Qualifier @Value
- create a plugin for PowerShell ISE
- 再读c++primer plus 003
- sql相同项求和
- Softmax &;&; Cross-entropy Error