传送门

矩阵很大,但是发现 $n$ 很小,从这边考虑,对于一个一堆小矩阵放在一起的情况

考虑把每一块单独考虑然后方案再乘起来

但是这些奇怪的东西很不好考虑

所以暴力一点,直接拆成一个个小块

但是这样我们还要考虑到小矩形的限制,设 $f[i][S]$ 表示现在考虑第 $i$ 个小块,小矩形的限制满足的状态为 $S$ 时的方案数

发现这些小块不会跨过矩形,维护每个小块的限制(即这个块能填的最大的数)$Mx$,以及这个小块填最大数时,能使哪些小矩形满足限制 ($P$)

设小块的面积为 $S$,那么如果下一小矩形不填最大数,则转移到 $f[i+1][S]$,贡献方案数为 $(Mx[i+1]-1)^{S[i+1]}$

如果下一小矩形填最大数,则转移到 $f[i+1][S|P[i+1]]$,贡献为总方案数-不填最大数的方案数$Mx[i+1]^{S[i+1]}\ -\ (Mx[i+1]-1)^{S[i+1]}$

然后就是奇奇怪怪的离散化和预处理了

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
typedef long long ll;
inline int read()
{
int x=,f=; char ch=getchar();
while(ch<''||ch>'') { if(ch=='-') f=-; ch=getchar(); }
while(ch>=''&&ch<='') { x=(x<<)+(x<<)+(ch^); ch=getchar(); }
return x*f;
}
const int N=,mo=1e9+;
int T,h,w,n,m;
int X1[N],X2[N],Y1[N],Y2[N],v[N];
int xp[N],yp[N],tx,ty,tot;
int tmp[N],t;
int f[N][N],S[N],Mx[N],P[N];
inline bool pd(int x,int y,int k) { return x>=X1[k]&&x<=X2[k]&&y>=Y1[k]&&y<=Y2[k]; }
//判断以(x,y)右上角的小块是否在矩形k中,因为离散化后小块不可能跨过矩形所以可以这样判断
inline int ksm(int x,int y)
{
int res=;
while(y)
{
if(y&) res=1ll*res*x%mo;
x=1ll*x*x%mo; y>>=;
}
return res;
}
int main()
{
//以下默认往右为大,往上为大
T=read();
while(T--)
{
memset(f,,sizeof(f)); memset(P,,sizeof(P));
xp[tx=]=; yp[ty=]=; tot=;
h=read(),w=read(),m=read(),n=read();
for(int i=;i<=n;i++)
{
X1[i]=read(),Y1[i]=read(),X2[i]=read(),Y2[i]=read(),v[i]=read();
xp[++tx]=X1[i]-,yp[++ty]=Y1[i]-;//注意-1,边界很重要,左下弄成开区间很重要!
xp[++tx]=X2[i],yp[++ty]=Y2[i];
}
xp[++tx]=h; yp[++ty]=w;
sort(xp+,xp+tx+); sort(yp+,yp+ty+);
for(int i=;i<=tx;i++) tmp[i]=xp[i]; t=tx; tx=;
for(int i=;i<=t;i++) if(i==||tmp[i]!=tmp[i-]) xp[++tx]=tmp[i];//离散化
for(int i=;i<=ty;i++) tmp[i]=yp[i]; t=ty; ty=;
for(int i=;i<=t;i++) if(i==||tmp[i]!=tmp[i-]) yp[++ty]=tmp[i];//离散化
for(int i=;i<=tx;i++)
for(int j=;j<=ty;j++)
{
tot++; Mx[tot]=m;
S[tot]=(xp[i]-xp[i-])*(yp[j]-yp[j-]);//小的边界是不包含的,即区间是左开右闭的,下开上闭的
for(int k=;k<=n;k++)
if(pd(xp[i],yp[j],k)) Mx[tot]=min(Mx[tot],v[k]);//处理Mx
for(int k=;k<=n;k++)
if(pd(xp[i],yp[j],k) && Mx[tot]==v[k])//处理P
P[tot]|=(<<k-);
}
int mx=(<<n)-; f[][]=;//DP
for(int i=;i<tot;i++)
{
int t1=ksm(Mx[i+]-,S[i+]),t2=(ksm(Mx[i+],S[i+])-t1+mo)%mo;
for(int j=;j<=mx;j++)
{
if(!f[i][j]) continue;
f[i+][j|P[i+]]=(f[i+][j|P[i+]]+1ll*f[i][j]*t2%mo)%mo;//此块填最大数
f[i+][j]=(f[i+][j]+1ll*f[i][j]*t1%mo)%mo;//此块不填最大数
}
}
printf("%d\n",f[tot][mx]);
}
}

P3813 [FJOI2017]矩阵填数的更多相关文章

  1. P3813 [FJOI2017]矩阵填数(组合数学)

    P3813 [FJOI2017]矩阵填数 shadowice1984说:看到计数想容斥........ 这题中,我们把图分成若干块,每块的最大值域不同 蓝后根据乘法原理把每块的方案数(互不相干)相乘. ...

  2. [luogu P3813] [FJOI2017] 矩阵填数 解题报告 (容斥原理)

    题目链接: https://www.luogu.org/problemnew/show/P3813 题目: 给定一个 h*w的矩阵,矩阵的行编号从上到下依次为 1..h,列编号从左到右依次1..w. ...

  3. [FJOI2017]矩阵填数——容斥

    参考:题解 P3813 [[FJOI2017]矩阵填数] 题目大意: 给定一个 h∗w 的矩阵,矩阵的行编号从上到下依次为 1...h ,列编号从左到右依次 1...w . 在这个矩阵中你需要在每个格 ...

  4. [BZOJ5010][FJOI2017]矩阵填数(状压DP)

    5010: [Fjoi2017]矩阵填数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 90  Solved: 45[Submit][Status][ ...

  5. bzoj5010: [Fjoi2017]矩阵填数

    Description 给定一个 h*w 的矩阵,矩阵的行编号从上到下依次为 1..h,列编号从左到右依次1..w.在这个矩阵中你需要在每 个格子中填入 1..m 中的某个数.给这个矩阵填数的时候有一 ...

  6. bzoj 5010: [Fjoi2017]矩阵填数

    Description 给定一个 h*w 的矩阵,矩阵的行编号从上到下依次为 1..h,列编号从左到右依次1..w.在这个矩阵中你需要在每 个格子中填入 1..m 中的某个数.给这个矩阵填数的时候有一 ...

  7. BZOJ5010 FJOI2017矩阵填数(容斥原理)

    如果只考虑某个子矩阵的话,其最大值为v的方案数显然是vsize-(v-1)size.问题在于处理子矩阵间的交叉情况. 如果两个交叉的子矩阵所要求的最大值不同,可以直接把交叉部分划给所要求的最大值较小的 ...

  8. 【BZOJ】5010: [Fjoi2017]矩阵填数

    [算法]离散化+容斥原理 [题意]给定大矩阵,可以每格都可以任意填1~m,给定n个子矩阵,要求满足子矩阵内的最大值为vi,求方案数. n<=10,h,w<=1w. [题解] 此题重点之一在 ...

  9. [FJOI2017]矩阵填数

    [Luogu3813] [LOJ2280] 写得很好的题解 \(1.\)离散化出每一块内部不互相影响的块 \(2.\)\(dp[i][j]\)为前 \(i\) 种重叠块其中有 \(j\) 这些状态的矩 ...

随机推荐

  1. c++原型模式(Prototype)

    原型模式是通过已经存在的对象的接口快速方便的创建新的对象. #include <iostream> #include <string> using namespace std; ...

  2. Angular25 组件的生命周期钩子

    1 生命周期钩子概述 组件共有9个生命周期钩子 1.1 生命周期的执行顺序 技巧01:测试时父组件传递对子组件的输入属性进行初始化操作 import { Component, Input, Simpl ...

  3. 如何在Django模型中管理并发性 orm select_for_update

    如何在Django模型中管理并发性 为单用户服务的桌面系统的日子已经过去了 - 网络应用程序现在正在为数百万用户提供服务,许多用户出现了广泛的新问题 - 并发问题. 在本文中,我将介绍在Django模 ...

  4. Eclipse中新建applet 错误

    出现的问题:  “错误,请单击以获取详细信息” Java Plug-in 1.6.0_45 使用 JRE 版本 1.6.0_45-b06 Java HotSpot(TM) Client VM 用户主目 ...

  5. Java IO输入输出流 FileWriter 字符流

    字节缓冲流 //为什么要使用包装流,使用包装流是为了提高读写操作的性能. public class Packing_flowDemo { public static void main(String[ ...

  6. DELPHI 调用系统 ADO 配置窗体 提高软件易用性

    最近DELPHI好像不太景气哦,把自己的代码拿出来晒晒.高手别喷哦. 直接上代码 implementation uses AdoConEd; var  saveconnstr:string; proc ...

  7. 将DataTable进行分页并生成新的DataTable

    /// <summary> /// 将DataTable进行分页并生成新的DataTable /// </summary> /// <param name="d ...

  8. C#字符串string的常用使用方法(转载)

    1--->字符串的声明: 1.string s=new string(char[] arr)     //根据一个字符数组声明字符串,即将字符字组转化为字符串. 2.string s=new s ...

  9. PLSQL Developer连接远程Oracle

    注:内容来网络 (一)不安装客户端的解决办法. 第一种方法: 1.在安装ORACLE服务器的机器上搜索下列文件, oci.dll ocijdbc10.dll ociw32.dll orannzsbb1 ...

  10. eclipse操作

    1.手动添加组件源码 2.源码阅读技巧 选择类Ctrl+T(Quick Type Hierarchy),查看该类的继承关系: 选择方法Ctrl+Alt+H(Open Call Hierarchy),查 ...