传送门

无比毒瘤的dp题,而且伪装地好像很可做的样子

考场上我给它氪了差不多一个小时最后还是只能扔了个20pts状压走人

以下思路基本均来源于题解:

对于此题,题面中三个限制条件:

(1)第 i 行第 1~li 列恰好有 1 个 1。 (li+1到ri-1不能放1)

(2)第 i 行第 ri~m 列恰好有 1 个 1。

(3)每列至多有 1 个 1。

根本注意不到注意到条件(3)相对比较好转移,所以此题考虑dp列而不是dp行

那么考虑从中选择一列该如何转移

显然每列只有放1或不放1两种情况,但条件(1)(2)在干扰我们转移

所以尝试消除干扰: 仅考虑单一区间(以右区间为例)

显然只有左端点到达这一列的区间需要被考虑

那么令\(dp[i][j]\)为从左到右第i列,所有跨越第i列的右区间中有j个已放过1

对于不放1的情况,\(dp[i][j] += dp[i-1][j]\)

对于放1,先定义数组\(cntl[i]\)为第i列及其左侧左区间的右端点的个数,\(cntr[i]\)同理

则第i列可以放1的左端点有\(cntr[i]-(j-1)\)个,所以\(dp[i][j] += dp[i-1][j-1]\times (cntr[i]-(j-1))\)

上面仅考虑了跨越第i列的右区间的方案数,那么下面处理左区间

首先定义\(f[i][j]\)时j指的是「所有跨越第i列的右区间中有j个已放过1」

那就不必考虑第i列的1应该给左区间还是右区间了

对于这种类似左右两边对抗的方案数dp

(就是说类似只能 在某条线以左/以右/以此线为分界左右同时 进行某种操作)

(为什么我一想到这就想起alpha-beta对抗搜索啊,好像将答案区间划分的思路差不多?)

考虑拆分区间,拆分出对左侧产生影响的右区间

则其对左区间影响已知,可计算出左区间

对于此题,i列左侧总共能放\(i-(j-1)\)个1,需要放\(cntl[i]-cntl[i-1]\)个

这个\(cntl[i]-cntl[i-1]\)其实是有多少个左区间在i位置结束,即「新增加的必须放1的区间个数」

是在满足i列以前必须放1的区间都满足的条件下转移

\(cntl[i]-cntl[i-1]\)个1放到\(i-(j-1)\)个行中,考虑不同方案,应该是排列数

所以\(dp[i][j] *= A^{cntl[i]-cntl[i-1]}_{i-(j-1)}\)

就可以转移了

一大坑点: 这类n, m混杂的题一定要分清n, m!我预处理阶乘逆元的时候习惯性打了i<=n直接调了一晚上最后还是战神帮忙指出来的谢谢战神小可爱啦大雾逃

Code:

#include <bits/stdc++.h>
using namespace std;
#define INF 0x3f3f3f3f
#define N 3010
#define ll long long
#define ld long double
#define usd unsigned
#define ull unsigned long long
//#define int long long #define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf, 1, 1<<21, stdin)), p1==p2?EOF:*p1++)
char buf[1<<21], *p1=buf, *p2=buf;
inline int read() {
int ans=0, f=1; char c=getchar();
while (!isdigit(c)) {if (c=='-') f=-f; c=getchar();}
while (isdigit(c)) {ans=(ans<<3)+(ans<<1)+(c^48); c=getchar();}
return ans*f;
} int n, m;
int l[N], r[N], lcnt[N], rcnt[N];
ll dp[N][N], fac[N], inv[N];
const ll mod=998244353; inline ll A(ll n, ll k) {return fac[n]*inv[n-k]%mod;}
inline ll md(ll a) {return a>=mod?a-mod:a;} signed main()
{
#ifdef DEBUG
freopen("1.in", "r", stdin);
#endif n=read(); m=read();
if ((n<<1)>m) {puts("0"); return 0;}
for (int i=1; i<=n; ++i) l[i]=read(), r[i]=read();
sort(l+1, l+n+1); sort(r+1, r+n+1);
fac[0]=fac[1]=1; inv[0]=inv[1]=1;
for (int i=2; i<=m; ++i) fac[i]=fac[i-1]*i%mod;
for (int i=2; i<=m; ++i) inv[i]=(mod-mod/i)*inv[mod%i]%mod;
for (int i=2; i<=m; ++i) inv[i]=inv[i-1]*inv[i]%mod;
for (int i=1,p=1; i<=m; ++i) {rcnt[i]=rcnt[i-1]; while (p<=n&&r[p]==i) ++rcnt[i],++p;}
for (int i=1,p=1; i<=m; ++i) {lcnt[i]=lcnt[i-1]; while (p<=n&&l[p]==i) ++lcnt[i],++p;}
dp[0][0]=1;
for (int i=1; i<=m; ++i) {
dp[i][0] = dp[i-1][0]*A(i-lcnt[i-1], lcnt[i]-lcnt[i-1])%mod;
for (int j=1; j<=min(i, n); ++j)
dp[i][j] = md(dp[i-1][j]+dp[i-1][j-1]*max(rcnt[i]-(j-1), 0)%mod)*A(i-j-lcnt[i-1], lcnt[i]-lcnt[i-1])%mod; //, cout<<"A: "<<dp[i][j]<<endl;
}
printf("%lld\n", dp[m][n]); return 0;
}

题解 matrix的更多相关文章

  1. 算法与数据结构基础 - 数组(Array)

    数组基础 数组是最基础的数据结构,特点是O(1)时间读取任意下标元素,经常应用于排序(Sort).双指针(Two Pointers).二分查找(Binary Search).动态规划(DP)等算法.顺 ...

  2. [LeetCode 题解] Spiral Matrix

    前言 [LeetCode 题解]系列传送门: http://www.cnblogs.com/double-win/category/573499.html 题目链接 54. Spiral Matrix ...

  3. 【题解】Sonya and Matrix Beauty [Codeforces1080E]

    [题解]Sonya and Matrix Beauty [Codeforces1080E] 传送门:\(Sonya\) \(and\) \(Matrix\) \(Beauty\) \([CF1080E ...

  4. 【题解】#6622. 「THUPC 2019」找树 / findtree(Matrix Tree+FWT)

    [题解]#6622. 「THUPC 2019」找树 / findtree(Matrix Tree+FWT) 之前做这道题不理解,有一点走火入魔了,甚至想要一本近世代数来看,然后通过人类智慧思考后发现, ...

  5. [题解]UVa 11082 Matrix Decompressing

    开始眨眼一看怎么也不像是网络流的一道题,再怎么看也觉得像是搜索.不过虽然这道题数据范围很小,但也不至于搜索也是可以随随便便就可以过的.(不过这道题应该是special judge,因为一题可以多解而且 ...

  6. [LeetCode]题解(python):074-Search a 2D Matrix

    题目来源 https://leetcode.com/problems/search-a-2d-matrix/ Write an efficient algorithm that searches fo ...

  7. [LeetCode]题解(python):059-Spiral Matrix II

    题目来源 https://leetcode.com/problems/spiral-matrix-ii/ Given an integer n, generate a square matrix fi ...

  8. [LeetCode]题解(python):054-Spiral Matrix

    题目来源 https://leetcode.com/problems/spiral-matrix/ Given a matrix of m x n elements (m rows, n column ...

  9. [LeetCode]题解(python):073-Set Matrix Zeroes

    题目来源: https://leetcode.com/problems/set-matrix-zeroes/ 题意分析: 输入一个m×n矩阵,如果出现有0,那么将对应的行和列都变成0. 题目思路: 简 ...

随机推荐

  1. mysql 索引介绍与运用

    索引 (1)什么是索引? 是一种提升查询速度的 特殊的存储结构. 它包含了对数据表里的记录的指针,类似于字典的目录. 当我们添加索引时会单独创建一张表来去存储和管理索引,索引比原数据大,会占用更多的资 ...

  2. SHELL 变量引用

    shell变量的引用非常重要,运用技巧灵活多变 变量的引用主要包含四类:双引号引用.单引号引用.反引号引用.反斜线引用 " " 双引号 屏蔽除美元符号$.反引号( ` )和反斜线( ...

  3. win10 sql2008r2网页不能使用数据,需要开启端口1433

    1.打开sql server configuratiton Manager 2.sqlserver网络配置--SQLEXPRESS的协议:都启用 3.双击TCP/IP:选择"IP地址&quo ...

  4. HTML表单__表单元素属性

    看完"HTML表单__表单元素"那一节的同学会发现,同是input标签,type属性值不一样的时候,input类型完全不一样.type就是input的一个属性,除type之外,还有 ...

  5. 10分钟系列:NetCore3.1+EFCore三步快速完成数据库交互

    前言 做程序开发,不管是什么语言什么数据库,其中的ORM(对象关系映射)是必不可少的,但是不管选择哪一种ORM,都需要了解其中的运行机制,配置帮助类等等. 所以很多ORM都开始进行升级封装,我们只需要 ...

  6. 两万字Vue.js基础学习笔记

    Vue.js学习笔记 目录 Vue.js学习笔记 ES6语法 1.不一样的变量声明:const和let 2.模板字符串 3.箭头函数(Arrow Functions) 4. 函数的参数默认值 5.Sp ...

  7. Java-数组有关

    1.复制数组 复制数组主要有三类方法: 1.使用循环语句逐个赋值数组元素 2.使用System类中的静态方法arraycopy 3.使用clone方法复制数组 对于2,详述如下: arraycopy( ...

  8. 网络损伤仪WANsim中的时延的不同模型

    网络损伤仪WANsim中的3种时延模型 时延指的是报文从网络的一端到达另一端所花费的时间. 网络损伤仪WANsim中为用户提供了3种时延损伤的模型.常量模型.均匀分布.正态分布. 这3种模型按照各自的 ...

  9. java 日期字符串互相转换

    一.把日期转换成字符串 //获取当前时间  Date date = new Date();   //打印date数据类型  System.out.println(date.getClass().get ...

  10. PostgreSQL-WITH AS短语

    WITH提供了一种方式来书写在一个大型查询中使用的辅助语句.这些语句通常被称为公共表表达式或CTE,它们可以被看成是定义只在一个查询中存在的临时表.在WITH子句中的每一个辅助语句可以是一个SELEC ...