题目链接:http://codeforces.com/problemset/problem/372/B

题意:

  给你一个n*m的01矩阵(1 <= n,m <= 40)。

  然后有t组询问(a,b,c,d),问你:

    在以(a,b)为左上角,以(c,d)为左下角,围成的矩形范围中,有多少全是0的矩形。

题解:

  这题是dp套dp……

  首先解决dp1:

    表示状态:

      f[a][b][c][d] = Rectangles number

      表示左上角为(a,b),右下角的范围在(c,d)以内的全0矩形的个数。

    如何转移:

      f[a][b][c][d] = f[a][b][c-1][d] + f[a][b][c][d-1] - f[a][b][c-1][d-1] + check(a,b,c,d)

      简单的容斥原理。

      其中,如果左上角为(a,b),右下角为(c,d)的矩形中全是0,则check(a,b,c,d)为1,否则为0(要用到二维前缀和)。

    边界条件:

      set f = 0

    复杂度O(N^4)。

  然后解决dp2:

    表示状态:

      dp[a][b][c][d] = Rectangles number

      表示在(a,b)和(c,d)围成的范围内的全0矩形个数。

      显然有:

        dp[a][b][c][d] = ∑ f[i][j][c][d] (a<=i<=c, b<=j<=d)

    如何转移:

      dp[a][b][c][d] = dp[a+1][b][c][d] + dp[a][b+1][c][d] - dp[a+1][b+1][c][d] + f[a][b][c][d]

      还是根据容斥原理,不过在这里a,b,c,d要倒着枚举。

    边界条件:

      set dp = 0

    复杂度O(N^4)。

  所以对于每一次询问,直接输出dp[a][b][c][d]即可。

  总复杂度O(N^4 + t)

AC Code:

 #include <iostream>
#include <stdio.h>
#include <string.h>
#define MAX_N 45 using namespace std; int n,m,t;
int v[MAX_N][MAX_N];
int s[MAX_N][MAX_N];
int f[MAX_N][MAX_N][MAX_N][MAX_N];
int dp[MAX_N][MAX_N][MAX_N][MAX_N]; void read()
{
cin>>n>>m>>t;
char c;
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
{
cin>>c;
v[i][j]=c-'';
}
}
} void cal_s()
{
memset(s,,sizeof(s));
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
{
s[i][j]=s[i][j-]+s[i-][j]-s[i-][j-]+v[i][j];
}
}
} int check(int a,int b,int c,int d)
{
int sum=s[c][d]-s[c][b-]-s[a-][d]+s[a-][b-];
return sum== ? : ;
} void cal_f()
{
memset(f,,sizeof(f));
for(int a=;a<=n;a++)
{
for(int b=;b<=m;b++)
{
for(int c=a;c<=n;c++)
{
for(int d=b;d<=m;d++)
{
f[a][b][c][d]=f[a][b][c-][d]+
f[a][b][c][d-]-
f[a][b][c-][d-]+
check(a,b,c,d);
}
}
}
}
} void cal_dp()
{
memset(dp,,sizeof(dp));
for(int a=n;a>=;a--)
{
for(int b=m;b>=;b--)
{
for(int c=n;c>=a;c--)
{
for(int d=m;d>=b;d--)
{
dp[a][b][c][d]=dp[a+][b][c][d]+
dp[a][b+][c][d]-
dp[a+][b+][c][d]+
f[a][b][c][d];
}
}
}
}
} void work()
{
cal_s();
cal_f();
cal_dp();
int a,b,c,d;
while(t--)
{
cin>>a>>b>>c>>d;
cout<<dp[a][b][c][d]<<endl;
}
} int main()
{
read();
work();
}

Codeforces 372B Counting Rectangles is Fun:dp套dp的更多相关文章

  1. Codeforces 372B Counting Rectangles is Fun

    http://codeforces.com/problemset/problem/372/B 题意:每次给出一个区间,求里面有多少个矩形 思路:预处理,sum[i][j][k][l]代表以k,l为右下 ...

  2. codeforces 979E(dp套dp)

    题意: 有n个点,编号为1~n.有的点颜色是黑色,有的点颜色是白色,有的点的颜色待涂.你还可以连一些边,但这些边一定是从小编号连到大编号的点. 对于一个确定的图,我们去统计有多少条路径满足“该路径经过 ...

  3. bzoj 3864: Hero meet devil [dp套dp]

    3864: Hero meet devil 题意: 给你一个只由AGCT组成的字符串S (|S| ≤ 15),对于每个0 ≤ .. ≤ |S|,问 有多少个只由AGCT组成的长度为m(1 ≤ m ≤ ...

  4. [模板] dp套dp && bzoj5336: [TJOI2018]party

    Description Problem 5336. -- [TJOI2018]party Solution 神奇的dp套dp... 考虑lcs的转移方程: \[ lcs[i][j]=\begin{ca ...

  5. luogu 4158 粉刷匠 dp套dp

    dp套dp 每个木板是个递推的dp,外部是个分组背包 #include<bits/stdc++.h> #define rep(i,x,y) for(register int i=x;i&l ...

  6. DP套DP

    DP套DP,就是将内层DP的结果作为外层DP的状态进行DP的方法. [BZOJ3864]Hero meet devil 对做LCS的DP数组差分后状压,预处理出转移数组,然后直接转移即可. tr[S] ...

  7. 【BZOJ3864】Hero meet devil DP套DP

    [BZOJ3864]Hero meet devil Description There is an old country and the king fell in love with a devil ...

  8. dp 套 dp扯谈

    1.[扯谈概念] \(dp\) 套 \(dp\) 其实也就是 \(dp\) . 这里就定义下面两个概念: 内层 \(dp\) 表示的是被套在里面的那个 \(dp\) 外层 \(dp\) 表示的是最外面 ...

  9. P4590-[TJOI2018]游园会【dp套dp】

    正题 题目链接:https://www.luogu.com.cn/problem/P4590 题目大意 给出一个长度为\(m\)的字符串\(s\). 对于每个\(k\in[0,m]\)求有多少个长度为 ...

随机推荐

  1. java使用命令wsimport构建WebService客户端

    wsimport -d d: -keep -extension -p com.demo.client http://192.168.33.3//RECEPTIONws.ASMX?WSDL 客户端:在J ...

  2. MongoDB入门学习(1)

    什么是MongoDB ? MongoDB 是由C++语言编写的,是一个基于分布式文件存储的开源数据库系统. 在高负载的情况下,添加更多的节点,可以保证服务器性能. MongoDB 旨在为WEB应用提供 ...

  3. MVC架构模式(转载)

    MVC架构模式 [概念理解] MVC模式和C++的实现 面向对象的设计模式是经验的总结,MVC思想是原先用于构建用户界面的.这篇文章主要论述了如何在新的Web应用领域中使用设计模式和MVC架构.文章首 ...

  4. php实现等比例不失真缩放上传图片的方法

    本文实例分析了php实现等比例不失真缩放上传图片的方法.分享给大家供大家参考,具体如下: 有时上传图片时因为图片太大了,不仅占用空间,消耗流量,而且影响浏(图片的尺寸大小不一).下面分享一种等比例不失 ...

  5. strpos 判断字符串是否存在

    strpos    中为什么要用逗号隔开的原因是因为   防止找出相匹配的中 ,  如   查找1    而数组中  存在  12  那么这个结果也是可以找出来的 ,分别在1 前后加个,  就是为了区 ...

  6. C# 中安全代码与不安全代码

    C# 中安全代码与不安全代码 P/Invoke 非托管代码需要在unsafe块中书写. using System; using System.Collections.Generic; using Sy ...

  7. COM对象模型

    COM对象内存布局,多继承是虚继承吗? 接口之间怎么切换? 1) 是普通的多继承,不是虚继承.因为父类接口只是含有纯虚函数,不含任何数据成员,所以问题不大. 2) QueryInterface可以用来 ...

  8. php序列化与反序列化

    php序列化简单来说就是 把复杂的数据类型压缩到一个字符串中php对象转换成字符串,反序列化就是把变量转换成对象 一般来说  当把这些序列化的数据放在URL中在页面之间会传递时,需要对这些数据调用ur ...

  9. SVN版本号控制软件-图片含义具体解释

    转载请注明出处:http://blog.csdn.net/zhuwentao2150/article/details/51195154 自己定义SVN图标显示风格 SVN的图标是能够自己定义风格的 右 ...

  10. .Net 开发Windows Service

    1.首先创建一个Windows Service 2.创建完成后切换到代码视图,代码中默认有OnStart和OnStop方法执行服务开启和服务停止执行的操作,下面代码是详细解释: using Syste ...