题目:https://loj.ac/problem/2719

首先要发现合法的充要条件是 | LDS | <=2 !

因为有没用的步数,说明一个元素先往左移、又往右移(不会先往右移再往左移,因为一旦往右移,说明它是前缀最大值,并且一直是),就说明它前面有一个比它大的、后面有一个比它小的,即有长度至少为 3 的 LDS 。

考虑 DP ,已填了前 i 个位置。注意到已经填过的最大的数最容易产生 LDS ,所以令 dp[ i ][ j ] 表示填了前 i 个位置、已填的最大数是 j 的方案数。

考虑第 i 个位置,如果填一个 < j 的数,只能填未填的最小数(这个发现很重要!),不然之后就会产生长度为 3 的 LDS 了;如果填一个 > j 的数,就没什么要注意的。

所以有 dp[ i ][ j ]*1 -> dp[ i+1 ][ k ] ( k>=j )

注意到系数全是 1 ,考虑用前缀和的思想来优化。就是 dp[ i ][ j ] 只向 dp[ i+1 ][ j ] 转移,然后 dp[ i ][ j ] 可以向 dp[ i ][ j+1 ] 转移。

如果 j = i ,考虑到没有 dp[ i+1 ][ j ] ,所以 dp[ i ][ j ] 向 dp[ i+1 ][ j+1 ] 转移即可。

每个状态画成二维格点,发现转移就是只能向右或向上、右下角一些位置不能走的 “从 ( 0 , 0 ) 走到 ( n , n ) ” 的方案数!可用卡特兰数。

那个斜着的边,把它改成横平竖直的即可。那么就是不能经过 y=x-2 这条直线。从 ( x1,y1 ) 走到 ( x2,y2 ) 的方案数就是 \( C_{x2-x1+y2-y1}^{x2-x1} - C_{x2-(y1+1)+y2-(x1-1)}^{x2-(y1+1)} \) 。

注意第0列是没有竖着的转移的。可以认为是从 ( 1 , 1 ) 走到 ( n , n ) 。

然后要保证字典序 > q 。考虑数位 DP 。

设当前做到第 i 位、之前都是贴着上界。设之前贴上界的已填数中最大值是 mx ,未填的最小值是 mn 。

第 i 位不贴上界,设填 k ,需要 k > q[ i ] ;还需要 k > mx ,因为若 k < mx ,又有 k > q[ i ] ,所以 mx , k , q[ i ] 会组成长度为 3 的 LDS 。

那么就是要 \( \sum\limits_{k>max(mx,q[i])} dp[i][k] \) ;这是一个后缀和,正好就是从 ( i , max( mx,q[ i ] )+1 ) 走到 ( n , n ) 的不经过 y=x-2 的方案数。

如果贴上界,需要满足 q[ i ] > mx || q[ i ] == mn ,不然会出现长度 >=3 的 LDS 。如果不满足的话,就没有从第 i 位往后贴上界的方案了,直接 break 掉。

注意预处理至 2*n 。

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
int rdn()
{
int ret=;bool fx=;char ch=getchar();
while(ch>''||ch<''){if(ch=='-')fx=;ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return fx?ret:-ret;
}
int Mx(int a,int b){return a>b?a:b;}
const int N=12e5+,mod=;//N=6e5*2!!!
int upt(int x){while(x>=mod)x-=mod;while(x<)x+=mod;return x;}
int pw(int x,int k)
{int ret=;while(k){if(k&)ret=(ll)ret*x%mod;x=(ll)x*x%mod;k>>=;}return ret;}
int n,a[N],jc[N],jcn[N]; bool vis[N];
void init()
{
int lm=12e5;
jc[]=;for(int i=;i<=lm;i++)jc[i]=(ll)jc[i-]*i%mod;
jcn[lm]=pw(jc[lm],mod-);
for(int i=lm-;i>=;i--)jcn[i]=(ll)jcn[i+]*(i+)%mod;
}
int C(int n,int m)
{
if(n<||m<||n<m)return ;
return (ll)jc[n]*jcn[m]%mod*jcn[n-m]%mod;
}
int cal(int x,int y)
{ return upt(C(*n-x-y,n-x)-C(*n-x-y,n-y-));}
int main()
{
freopen("inverse.in","r",stdin);
freopen("inverse.out","w",stdout);
int T=rdn(); init();
while(T--)
{
n=rdn(); for(int i=;i<=n;i++)a[i]=rdn();
memset(vis,,sizeof vis); int mn=,mx=,ans=;
for(int i=;i<=n;i++)
{
ans=upt(ans+cal(i,Mx(a[i],mx)+));
if(a[i]<mx&&a[i]!=mn)break;
mx=Mx(mx,a[i]); vis[a[i]]=;
while(i<n&&vis[mn])mn++;
}
printf("%d\n",ans);
}
return ;
}

LOJ 2719 「NOI2018」冒泡排序——模型转化的更多相关文章

  1. Loj #2719. 「NOI2018」冒泡排序

    Loj #2719. 「NOI2018」冒泡排序 题目描述 最近,小 S 对冒泡排序产生了浓厚的兴趣.为了问题简单,小 S 只研究对 *\(1\) 到 \(n\) 的排列*的冒泡排序. 下面是对冒泡排 ...

  2. loj 2719 「NOI2018」冒泡排序 - 组合数学

    题目传送门 传送门 题目大意 (相信大家都知道) 显然要考虑一个排列$p$合法的充要条件. 考虑这样一个构造$p$的过程.设排列$p^{-1}_{i}$满足$p_{p^{-1}_i} = i$. 初始 ...

  3. LOJ #2719. 「NOI2018」冒泡排序(组合数 + 树状数组)

    题意 给你一个长为 \(n\) 的排列 \(p\) ,问你有多少个等长的排列满足 字典序比 \(p\) 大 : 它进行冒泡排序所需要交换的次数可以取到下界,也就是令第 \(i\) 个数为 \(a_i\ ...

  4. LOJ 3056 「HNOI2019」多边形——模型转化+树形DP

    题目:https://loj.ac/problem/3056 只会写暴搜.用哈希记忆化之类的. #include<cstdio> #include<cstring> #incl ...

  5. LOJ #2721. 「NOI2018」屠龙勇士(set + exgcd)

    题意 LOJ #2721. 「NOI2018」屠龙勇士 题解 首先假设每条龙都可以打死,每次拿到的剑攻击力为 \(ATK\) . 这个需要支持每次插入一个数,查找比一个 \(\le\) 数最大的数(或 ...

  6. loj#2721. 「NOI2018」屠龙勇士

    题目链接 loj#2721. 「NOI2018」屠龙勇士 题解 首先可以列出线性方程组 方程组转化为在模p意义下的同余方程 因为不保证pp 互素,考虑扩展中国剩余定理合并 方程组是带系数的,我们要做的 ...

  7. 「NOI2018」冒泡排序

    「NOI2018」冒泡排序 考虑冒泡排序中一个位置上的数向左移动的步数 \(Lstep\) 为左边比它大的数的个数,向右移动的步数 \(Rstep\) 为右边比它大的数的个数,如果 \(Lstep,R ...

  8. loj#2718. 「NOI2018」归程

    题目链接 loj#2718. 「NOI2018」归程 题解 按照高度做克鲁斯卡尔重构树 那么对于询问倍增找到当前点能到达的高度最小可行点,该点的子树就是能到达的联通快,维护子树中到1节点的最短距离 s ...

  9. LOJ2719 「NOI2018」冒泡排序

    「NOI2018」冒泡排序 题目描述 最近,小S 对冒泡排序产生了浓厚的兴趣.为了问题简单,小 S 只研究对 1 到n 的排列的冒泡排序. 下面是对冒泡排序的算法描述. 输入:一个长度为n 的排列p[ ...

随机推荐

  1. sysbench - 数据库功能及性能测试工具

    sysbench 的 GitHub 参考资料 1.0 之后的版本使用方法跟之前的有所区别,下面所有内容基于 1.0.9 版本. 另外,为了方便管理测试,最好不要通过命令直接运行测试,而是写成脚本自动化 ...

  2. php Function ereg() is deprecated的解决方法

    PHP 5.3 ereg() 无法正常使用,提示“Function ereg() is deprecated Error”.问题根源是php中有两种正则表示方法,一个是posix,一个是perl,ph ...

  3. 巧用css内容生成

    1.      .box:before{content:"生成内容";}在.box内部的内容之前加上生成内容 2.       .box:after{content:"生 ...

  4. [Bzoj1051][HAOI2006]受欢迎的牛(tarjan)

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1051 由题意可知,被所有牛仰慕的牛之间也互相仰慕,则最后的答案一定是唯一的强连通分量,如 ...

  5. Laya2.0的转变

    之前一直用Laya1.x+TypeScript了,最近项目开始使用Laya2.0+AS3了 总结一下需要注意的一些事项,算是2种开发模式的区别与过渡吧 1.AS类的访问标识 必须是public,不写会 ...

  6. python学习第十一天列表的分片和运算

    列表的分片也叫切片,也就是从列表中取出一段赋值给另外一个变量,列表运算就是可以进行比较运算,连接运算,乘法运算等. 1,列表的分片 n1=[1,2,3,4,5,6,7,8,9] n2=[1:3] 包含 ...

  7. go web编程——实现一个简单分页器

    在go web编程中,当需要展示的列表数据太多时,不可避免需要分页展示,可以使用Go实现一个简单分页器,提供各个数据列表展示使用.具体需求:1. 可展示“首页”和“尾页”.2. 可展示“上一页”和“下 ...

  8. Spring学习笔记(14)——SpEL

    是什么 Spring表达式语言全称为"Spring Expression Language",缩写为"SpEL",类似于Struts2x中使用的OGNL表达式语 ...

  9. vue.js(7)--vue中的样式绑定

    vue中class样式与内联样式的绑定 <!DOCTYPE html> <html lang="en"> <head> <meta cha ...

  10. 前端对base64编码的理解,原生js实现字符base64编码

    目录 常见对base64的认知(不完全正确) 多问一个为什么,base64到底是个啥? 按照我们的思路实现一下 到这里基本就实现了,结果跟原生的方法打印的是一样的 下一次 @( 对于前端工程师来说ba ...