P2217 [HAOI2007]分割矩阵

Posted lltyyc

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了P2217 [HAOI2007]分割矩阵相关的知识,希望对你有一定的参考价值。

传送门

首先均方差公式: $\sigma = \sqrt\sum_i^K\frac(sum[i]-\barsum)^2n$

其中 $\barsum$ 为小矩阵的平均值,显然 $\barsum=\frac\sum_i^Ksum[i]K$

所以就是要最小化 $(sum[i]-\barsum)^2$

看到数据这么小,搜就完事了

直接 $dfs(xa,ya,xb,yb,k)$ 表示以 $(xa,ya)$ 为左下角,$(xb,yb)$ 为右上角的子矩阵内,切 $k$ 次后的 $(sum[i]-\barsum)^2$ 最小值

然后发现重复的状态很多,所以记忆化一下,稳了

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
typedef long long ll;
typedef double db;
inline int read()

    int x=0,f=1; char ch=getchar();
    while(ch<0||ch>9)  if(ch==-) f=-1; ch=getchar(); 
    while(ch>=0&&ch<=9)  x=(x<<1)+(x<<3)+(ch^48); ch=getchar(); 
    return x*f;

const int N=11,INF=1e9;
int n,m,K,sum[N][N];
db f[N][N][N][N][N],P;
bool vis[N][N][N][N][N];
inline db calc(int xa,int ya,int xb,int yb)  return sum[xb][yb]-sum[xa-1][yb]-sum[xb][ya-1]+sum[xa-1][ya-1]; 
db dfs(int xa,int ya,int xb,int yb,int k)

    if(xb-xa+yb-ya<k) return INF;
    db &T=f[xa][ya][xb][yb][k];
    if(vis[xa][ya][xb][yb][k]) return T;
    vis[xa][ya][xb][yb][k]=1; T=INF;
    if(!k)  T=(calc(xa,ya,xb,yb)-P)*(calc(xa,ya,xb,yb)-P); return T; 
    for(int i=0;i<k;i++)
        for(int j=xa;j<xb;j++)
            T=min(T, dfs(xa,ya,j,yb,i)+dfs(j+1,ya,xb,yb,k-i-1) );
    for(int i=0;i<k;i++)
        for(int j=ya;j<yb;j++)
            T=min(T, dfs(xa,ya,xb,j,i)+dfs(xa,j+1,xb,yb,k-i-1) );
    return T;

int main()

    n=read(),m=read(),K=read();
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++) sum[i][j]=sum[i-1][j]+sum[i][j-1]-sum[i-1][j-1]+read();
    P=1.0*sum[n][m]/K;
    printf("%.2lf\n",sqrt( dfs(1,1,n,m,K-1)/K ));
    return 0;

 

以上是关于P2217 [HAOI2007]分割矩阵的主要内容,如果未能解决你的问题,请参考以下文章

[HAOI2007]分割矩阵

[bzoj1048] [HAOI2007]分割矩阵

1048: [HAOI2007]分割矩阵

BZOJ-1048: [HAOI2007]分割矩阵 (记忆化搜索)

「二分答案 + 搜索」[HAOI2007]覆盖问题

[HAOI2007]理想的正方形