【做题】arc068_f-Solitaire——糊结论
把所有数字放入双端队列后,结果大概是这样一个排列:
\]
其中\(P_1\)是递减序列,\(P_2\)是递增序列。
我们以\(1\)所在的位置\(k\)分割最终的排列\(A\)。
其中前半部分,形象地讲是由两个递减序列交织在一起组成的。那两个递减序列分别是\(P_1\)的前缀和\(P_2\)的后缀,且至少有一个就是\(P_1\)(\(P_2\))本身。那么,前半部分满足这个性质:
可以划分为两个可为空的递减子序列。
显然这个性质难以直接判定。考虑将其转化。对于其中编号为\(i\)的元素,设\(j\)是\(i\)后面的第一个与\(A_i\)属于不同子序列的元素位置。我们不妨分类讨论一发:
- \(A_i > A_j\),因为两个子序列递减,有\(\forall j \in [i+1,k-1], \, A_i>A_j\)
- \(A_i < A_j\),因为两个子序列递减,有\(\forall j \in [1,i-1], \, A_i<A_j\)
因此,我们得出:
长度为\(n\)的最终排列\(A\)的前半部分可以划分为两个可为空的递减子序列$\implies \(
\) \forall i \in [1,k-1]$,下列条件至少满足一个:
- \(\forall j \in [i+1,k-1], \, A_i>A_j\),记为条件1
- \(\forall j \in [1,i-1], \, A_i<A_j\),记为条件2
我们发现,逆命题也是成立的,因为最长不上升子序列的长度小于等于2。(导弹拦截可经典了)于是我们就获得了一个判断方法。
但还有一个问题,即从\(1\)到\(n\)的排列中任取出两个元素不重复的递减序列\(D_1\),\(D_2\),它们组成的序列\(P\),不一定可以作为\(A\)的前半部分。我们还要保证\(D_1\)的最后一个元素和\(D_2\)的最后一个元素中的较大值大于后半部分的所有元素。依然要对此进行转化。
那么,对于我们已知的一个\(A\)的前半部分,我们尝试将其划分且最大化两个子序列最后一个元素的较大值,设它的位置是\(p\)。显然\(p\)后面没有比\(A_p\)更大的元素,否则能得到更优解。通过简单分类讨论,我们发现所有不满足条件2的元素都大于后半部分的所有元素。因此我们可以直接修改条件2:
\]
故我们得到了前半部分的判断方法。
后半部分的判断就简单多了。它是一个递减序列不断从前面或后面取出元素得到的,即每次取出的元素都是剩下的元素中最大(最小)的。容易得到且证明它合法的充要条件:
$ \forall i \in [k+1,n]$,下列条件至少满足一个:
- \(\forall j \in [i+1,n], \, A_i > A_j\)
- \(\forall j \in [i+1,n], \, A_i < A_j\)
我们终于简化了题意:
求有多少个长度为\(n\)的排列\(A\)满足:
- \(A_k = 1\)
- \(\forall i \in [1,k)\),\(A_i\)小于它前面的所有数或大于它后面的所有数。
- \(\forall i \in (k,n]\),\(A_i\)小于它后面的所有数或大于它后面的所有数。
剩下的就简单了。由递推式可知,后半部分的方案数就是\(2^{n-k-1}\)。而对于前半部分,我们设dp状态\(i,j\)表示当前有\(i\)个元素,最小的满足条件2的元素是其中第\(j\)个(不存在则为\(j+1\)),最终答案就是
\]
时间复杂度\(O(n^2)\)。
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int N = 2010, MOD = 1e9 + 7;
int dp[2][N],n,k,s,p,sum[N],ans,jc[N],inv[N];
typedef long long ll;
ll power(ll a,int b) {
ll res = 1;
while (b) {
if (b&1) (res *= a) %= MOD;
(a *= a) %= MOD;
b >>= 1;
}
return res;
}
ll comb(int a,int b) {
if (a < 0 || b < 0 || a < b) return 0;
return 1ll * jc[a] * inv[b] % MOD * inv[a-b] % MOD;
}
signed main() {
scanf("%lld%lld",&n,&k);
jc[0] = 1;
for (int i = 1 ; i <= n ; ++ i)
jc[i] = 1ll * jc[i-1] * i % MOD;
inv[n] = power(jc[n],MOD-2);
for (int i = n-1 ; i >= 0 ; -- i)
inv[i] = 1ll * inv[i+1] * (i+1) % MOD;
s = k-1;
p = 0;
sum[2] = 1;
sum[3] = -1;
for (int i = 1 ; i <= s ; ++ i) {
p ^= 1;
memset(dp[p],0,sizeof dp[p]);
for (int j = 1 ; j <= i+1 ; ++ j)
(sum[j] += sum[j-1]) %= MOD;
for (int j = 1 ; j <= i+1 ; ++ j)
(dp[p][j] += sum[j]) %= MOD, sum[j] = 0;
sum[i+2] = 0;
for (int j = 1 ; j <= i+1 ; ++ j) {
(sum[2] += dp[p][j]) %= MOD;
(sum[j+2] -= dp[p][j]) %= MOD;
}
}
for (int i = 1 ; i <= s + 1 ; ++ i) {
(ans += 1ll * comb(n-k+i-1,i-1) * dp[p][i] % MOD);
ans %= MOD;
}
if (k == 1) ans = 1;
if (n > k)
ans = 1ll * power(2,n-k-1) * ans % MOD;
ans = (ans + MOD) % MOD;
printf("%lld\n",ans);
return 0;
}
小结:推导结论实在是一个困难的过程,积极的心态和清晰的思路是不可或缺的。
【做题】arc068_f-Solitaire——糊结论的更多相关文章
- AtCoder Grand Contest 11~17 做题小记
原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-11-to-20.html UPD(2018-11-16): ...
- AtCoder Grand Contest 1~10 做题小记
原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-1-to-10.html 考虑到博客内容较多,编辑不方便的情 ...
- noip做题记录+挑战一句话题解?
因为灵巧实在太弱辽不得不做点noip续下命QQAQQQ 2018 积木大赛/铺设道路 傻逼原题? 然后傻逼的我居然检查了半天是不是有陷阱最后花了差不多一个小时才做掉我做过的原题...真的傻逼了我:( ...
- 从【BZOJ4173】谈做题技巧
题目描述 ----------------------------------------------------------------------------------------------- ...
- PKUWC/SC 做题笔记
去年不知道干了些啥,什么省选/营题都没做. 现在赶应该还来得及(?) 「PKUWC2018」Minimax Done 2019.12.04 9:38:55 线段树合并船新玩法??? \(O(n^2)\ ...
- 退役II次后做题记录
退役II次后做题记录 感觉没啥好更的,咕. atcoder1219 历史研究 回滚莫队. [六省联考2017]组合数问题 我是傻逼 按照组合意义等价于\(nk\)个物品,选的物品\(\mod k\) ...
- BJOI做题记录
BJOI做题记录 终于想起还要做一下历年省选题了2333 然而咕了的还是比做了的多2333 LOJ #2178. 「BJOI2017」机动训练 咕了. LOJ #2179. 「BJOI2017」树的难 ...
- csp退役前的做题计划1(真)
csp退役前的做题计划1(真) 因为我太菜了,所以在第一次月考就会退役,还是记录一下每天做了什么题目吧. 任务计划 [ ] Z算法(Z Algorithm) 9.28 [x] ARC061C たくさん ...
- POI做题笔记
POI2011 Conspiracy (2-SAT) Description \(n\leq 5000\) Solution 发现可拆点然后使用2-SAT做,由于特殊的关系,可以证明每次只能交换两个集 ...
随机推荐
- kail linux arp欺骗
首先连接wifi,进入内网 1,查看内网的存活主机 命令 fping -asg 192.168.1.0/24 (视不同环境而定,假设这里的路由器地址为 192.168.1.1) 也可利用其他 ...
- discuz用户组
非公众用户组当用户组设置为“非公众用户组”时,无论是以主用户组的形式,还是以扩展用户组的形式,均只能由管理员手工将用户加入本组. 公众用户组当本用户组设置为“公众用户组”,且用户当前所在的用户组被允许 ...
- Python - 3. Input and Output
from:http://interactivepython.org/courselib/static/pythonds/Introduction/InputandOutput.html Input a ...
- 创建一个简单的WCF程序
1.创建WCF服务库 打开VS2010,选择文件→新建→项目菜单项,在打开的新建项目对话框中,依次选择Visual C#→WCF→WCF服务库,然后输入项目名称(Name),存放位置(Location ...
- 量化交易-外汇交易-MetaTrader5
量化交易-外汇交易-MetaTrader5 外汇有充足的流动性, 7*24, 交易成本低,多空双向,外加杠杆,无人能控盘,有模拟盘,相当适合做量化交易练习积累经验. 第一,全球最大最公平的市场.外汇市 ...
- 新服务器上装java PHP环境有什么一键安装的方便的方法?一般都是怎么安装环境的?
新服务器上装java PHP环境有什么一键安装的方便的方法?一般都是怎么安装环境的? linode digitalocean都有很好的教程,下面是ubuntu和centos的两个教程连接. How ...
- 计蒜客--移除数组中的重复元素 (set)
给定一个升序排列的数组,去掉重复的数,并输出新的数组的长度. 例如:数组 A = \{1, 1, 2\}A={1,1,2},你的程序应该输出 22 即新数组的长度,新数组为 \{1, 2\}{1,2} ...
- flask框架----设置配置文件的几种方式
设置配置文件的几种方式 ==========方式一:============ app.config['SESSION_COOKIE_NAME'] = 'session_lvning' #这种方式要把所 ...
- VMware激活密钥
VMware 2017 v14.x 永久许可证激活密钥FF31K-AHZD1-H8ETZ-8WWEZ-WUUVACV7T2-6WY5Q-48EWP-ZXY7X-QGUWD 原文链接
- 通过 Java 线程堆栈进行性能瓶颈分析
改善性能意味着用更少的资源做更多的事情.为了利用并发来提高系统性能,我们需要更有效的利用现有的处理器资源,这意味着我们期望使 CPU 尽可能出于忙碌状态(当然,并不是让 CPU 周期出于应付无用计算, ...