\(\mathcal{Description}\)

  Link.

  有 \(n\) 个牛棚,大小为 \(t_{1..n}\),\(n\) 头奶牛,大小为 \(s_{1..n}\),奶牛只能住进不小于自己的牛棚,每个牛棚最多住一头奶牛。求满足不能让更多奶牛住进牛棚的安排方案数,答案对 \((10^9+7)\) 取模。

  \(n\le3\times10^3\)。

\(\mathcal{Solution}\)

  把 \(s\) 和 \(t\) 倒一块儿升序排序,大小相同奶牛优先。那么相当于奶牛只能住进自己后面的某个房间,转化成了熟悉的括号问题。

  考虑一个方案合法的条件:最左侧失配的左括号(奶牛)在最右侧失配的右括号(牛棚)右侧(否则它们就能够再匹配了)。

  接下来的 DP 就比较平凡啦,令 \(f(i,j,0/1)\) 表示考虑了序列前 \(i\) 个元素,有 \(j\) 个左括号预定了接下来的右括号和它匹配,出现 / 未出现“最左侧失配的左括号”时的方案数。分第 \(i\) 个位置是奶牛还是牛棚转移(\(\longleftarrow\) 表示前者被后者贡献):

  • 奶牛转移:

    • 预定后方房间,\(f(i,j,0/1)\longleftarrow f(i-1,j-1,0/1)\);
    • 成为最左侧失配,\(f(i,j,1)\longleftarrow f(i-1,j,0)\);
    • 再次失配,\(f(i,j,1)\longleftarrow f(i-1,j,1)\)。
  • 牛棚转移:
    • 去满足某个房间预定,\(f(i,j,0/1)\longleftarrow (j+1)\cdot f(i-1,j+1,0/1)\);
    • 失配,\(f(i,j,0)\longleftarrow f(i-1,j,0)\)。

  转移最后一步体现了合法限制。最后答案显然是 \(f(2n,0,0)+f(2n,0,1)\)。

  最终,\(\mathcal O(n^2)\) 解决问题。

\(\mathcal{Code}\)

  代码中最后一维 \(0/1\) 状态是反过来的 awa。

/* Clearink */

#include <cstdio>
#include <algorithm> #define rep( i, l, r ) for ( int i = l, rpbound##i = r; i <= rpbound##i; ++i )
#define per( i, r, l ) for ( int i = r, rpbound##i = l; i >= rpbound##i; --i ) typedef std::pair<int, int> PII;
#define fi first
#define se second const int MAXN = 3e3, MOD = 1e9 + 7;
int n, f[MAXN * 2 + 5][MAXN + 5][2]; // !!!
PII s[MAXN * 2 + 5]; inline int mul( const long long a, const int b ) { return a * b % MOD; }
inline int sub( int a, const int b ) { return ( a -= b ) < 0 ? a + MOD : a; }
inline int add( int a, const int b ) { return ( a += b ) < MOD ? a : a - MOD; }
inline void addeq( int& a, const int b ) { ( a += b ) >= MOD && ( a -= MOD ); } int main() {
// freopen ( "sleeping.in", "r", stdin );
// freopen ( "sleeping.out", "w", stdout );
scanf( "%d", &n );
rep ( i, 1, n ) scanf( "%d", &s[i].fi ), s[i].se = 0;
rep ( i, n + 1, n << 1 ) scanf( "%d", &s[i].fi ), s[i].se = 1;
std::sort( s + 1, s + ( n << 1 | 1 ) );
f[0][0][1] = 1;
rep ( i, 1, n << 1 ) {
if ( !s[i].se ) rep ( j, 0, n ) { // cow.
if ( j ) {
addeq( f[i][j][1], f[i - 1][j - 1][1] );
addeq( f[i][j][0], f[i - 1][j - 1][0] );
}
addeq( f[i][j][0], add( f[i - 1][j][0], f[i - 1][j][1] ) );
} else rep ( j, 0, n ) { // house.
f[i][j][0] = mul( j + 1, f[i - 1][j + 1][0] );
f[i][j][1] = add( mul( j + 1, f[i - 1][j + 1][1] ),
f[i - 1][j][1] );
}
}
printf( "%d\n", add( f[n << 1][0][0], f[n << 1][0][1] ) );
return 0;
}

Solution -「USACO 2020.12 P」Sleeping Cows的更多相关文章

  1. Solution -「USACO 2020.12 P」Spaceship

    \(\mathcal{Description}\)   Link.   Bessie 在一张含 \(n\) 个结点的有向图上遍历,站在某个结点上时,她必须按下自己手中 \(m\) 个按钮中处于激活状态 ...

  2. Solution -「SV 2020 Round I」SA

    \(\mathcal{Description}\)   求出处 owo.   给定一个长度为 \(n\),仅包含小写字母的字符串 \(s\),问是否存在长度为 \(n\),仅包含小写字母的字符串 \( ...

  3. Solution -「SV 2020 Round I」「SRM 551 DIV1」「TC 12141」SweetFruits

    \(\mathcal{Description}\)   link.   给定 \(n\) 个水果,每个结点可能有甜度 \(v_i\),或不甜(\(v_i=-1\)).现在把这些水果串成一棵无根树.称一 ...

  4. [LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞

    [LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞 试题描述 到河北省 见斯大林 / 在月光下 你的背影 / 让我们一起跳舞吧 うそだよ~ 河北省怎么可能有 Stalin. ...

  5. Solution -「2020.12.26」 模拟赛

    0x00 前言 一些吐槽. 考得很变态诶,看每道题平均两秒的时限就知道了... T1 降智了想到后缀懒得打. T2 口胡了假优化,结果和暴力分一样?? T3 黑题还绑点?? \(50 + 80 + 0 ...

  6. Solution -「ZJOI 2020」「洛谷 P6631」序列

    \(\mathcal{Description}\)   Link.   给定一个长为 \(n\) 的非负整数序列 \(\lang a_n\rang\),你可以进行如下操作: 取 \([l,r]\),将 ...

  7. Solution -「JOISC 2020」「UOJ #509」迷路的猫

    \(\mathcal{Decription}\)   Link.   这是一道通信题.   给定一个 \(n\) 个点 \(m\) 条边的连通无向图与两个限制 \(A,B\).   程序 Anthon ...

  8. Solution -「NOI 2020」「洛谷 P6776」超现实树

    \(\mathcal{Description}\)   Link.   对于非空二叉树 \(T\),定义 \(\operatorname{grow}(T)\) 为所有能通过若干次"替换 \( ...

  9. Solution -「FJWC 2020」人生

    \(\mathcal{Description}\)   OurOJ.   有 \(n\) 个结点,一些结点有染有黑色或白色,其余待染色.将 \(n\) 个结点染上颜色并连接有向边,求有多少个不同(结点 ...

随机推荐

  1. C/C++避免头文件重复包含的方法

    C/C++避免头文件重复包含的方法 1. #ifndef 2. #pragma once 3. 混合使用 在实际的编程过程中,因为会使用多个文件,所以在文件中不可避免的要引入一些头文件,这样就可能会出 ...

  2. Linux上天之路(十)之Linux磁盘管理

    主要内容 磁盘介绍 磁盘管理 磁盘限额 逻辑卷管理 磁盘阵列 1. 磁盘介绍 硬盘最基本的组成部分是由坚硬金属材料制成的涂以磁性介质的盘片,不同容量硬盘的盘片数不等.每个盘片有两面,都可记录信息.盘片 ...

  3. Go语言系列之标准库ioutil

    ioutil标准库中提供了一些常用.方便的IO操作函数 一.相关方法 func ReadAll(r io.Reader) ([]byte, error) func ReadDir(dirname st ...

  4. 服务性能监控系列之Metrics

    Metrics是一个提供服务性能检测工具的Java类库,它提供了功能强大的性能指标工具库用于度量生产环境中的各关键组件性能. 度量类型 Metrics提供了以下几种基本的度量类型: Gauge:用于提 ...

  5. PHP代码审计之create_function()函数

    0x00 create_function()简介 适用范围:PHP 4> = 4.0.1,PHP 5,PHP 7 功能:根据传递的参数创建匿名函数,并为其返回唯一名称. 语法: 1 create ...

  6. Git使用:版本回退

    在Git中,我们可以用 git log命令查看我们修改的历史记录 C:\Users\Administrator\Documents\GitHub\learngit [master]> git l ...

  7. Solon 开发,四、Bean 扫描的三种方式

    Solon 开发 一.注入或手动获取配置 二.注入或手动获取Bean 三.构建一个Bean的三种方式 四.Bean 扫描的三种方式 五.切面与环绕拦截 六.提取Bean的函数进行定制开发 七.自定义注 ...

  8. C# 给PDF文档设置过期时间

    我们可以给一些重要文档或者临时文件设置过期时间和过期信息提示来提醒读者或管理者文档的时效性,并及时对文档进行调整.更新等.下面,分享通过C#程序代码来给PDF文档设置过期时间的方法. 引入dll程序集 ...

  9. cesium加载gltf模型点击以及列表点击定位弹窗

    前言 cesium 官网的api文档介绍地址cesium官网api,里面详细的介绍 cesium 各个类的介绍,还有就是在线例子:cesium 官网在线例子,这个也是学习 cesium 的好素材. 之 ...

  10. 快速删除IDEA/WebStrom/Rider中的代码空行

    使用替换 ^\s*\n 并打开正则匹配模式   Visual Studio中未测试,大家可以去试一试