A. 排序

题目描述

输入格式

输出格式

一行,一个整数,表示可以将数组A从小到大排序的不同的操作序列的个数。

样例

样例输入

3
7 8 5 6 1 2 4 3

样例输出

6

数据范围与提示

对于30%的数据,1<=N<=4; 对于全部的数据,1<=N<=12。

一个伪装成计数题的大搜索…

首先要知道一个结论:对于一个操作序列,如果他是合法的,那么他的全排列都是合法的,所以从小到大搜索,同时记录操作数k。

对于第x次操作,把序列分成2x-1段,暴力搜索有几段不满足a[i]=a[i-1]+1(注意这里并不只是要求满足递增),如果大于两段,显然无解;

如果没有,则第x操作无需进行(即对答案无贡献),直接向下递归即可,因为如果此时进行第x次操作,会将顺序打乱,之后的操作并不能使之还原。

如果有1段,则将这一段从中间断开,暴力交换,如果仍不满足则无解,满足则继续向下递归,k++;

如果有2端,则有四种情况,暴力交换向下递归即可,注意回溯时要还原。

结束条件:a[i]=i,答案为k!(即A(k,k));

不过这题看起来复杂度好高啊,居然能跑的这么快…

#include<iostream>
#include<cstdio>
#include<cmath>
#define LL long long
using namespace std;
int n,m=1;
int a[5010];
LL jc[15]; bool end()
{
for(int i=1;i<=m;i++)if(a[i]!=i)return 0;return 1;
}
void swapp(int l1,int r1,int l2,int r2)
{
for(int i=l1,j=0;i<=r1;i++,j++)
swap(a[i],a[l2+j]);
}
LL dfs(int x,int k)
{
if(end())return jc[k];
int len=pow(2,x),num=pow(2,n-x),te=0;
int pd=0,d1=0,d2=0;
for(int i=1;i<=num;i++)
for(int j=(i-1)*len+2;j<=i*len;j++)
if(a[j]!=a[j-1]+1) pd++,d1?d2=i:d1=i;
if(pd>2)return 0;
if(!pd)return dfs(x+1,k);
if(pd==1)
{
int l=(d1-1)*len+1,r=(d1)*len;bool pp=0;LL res=0;
swapp(l,(l+r)>>1,(l+r)/2+1,r);
for(int i=l+1;i<=r;i++)
if(a[i]<a[i-1]){pp=1;break;}
if(!pp)res=dfs(x+1,k+1);
swapp(l,(l+r)>>1,(l+r)/2+1,r);
return res;
}
if(pd==2)
{
int l1=(d1-1)*len+1,r1=(d1)*len,l2=(d2-1)*len+1,r2=(d2)*len;
int mid1=(l1+r1)>>1,mid2=(l2+r2)>>1;
LL res=0;
swapp(l1,mid1,l2,mid2);
{
int pp=0;
for(int i=l1+1;i<=r1;i++)
if(a[i]<a[i-1]){pp=1;break;}
for(int i=l2+1;i<=r2;i++)
if(a[i]<a[i-1]||pp){pp=1;break;}
if(!pp) res+=dfs(x+1,k+1);
}
swapp(l1,mid1,l2,mid2);
swapp(l1,mid1,mid2+1,r2);
{
int pp=0;
for(int i=l1+1;i<=r1;i++)
if(a[i]<a[i-1]){pp=1;break;}
for(int i=l2+1;i<=r2;i++)
if(a[i]<a[i-1]||pp){pp=1;break;}
if(!pp) res+=dfs(x+1,k+1);
}
swapp(l1,mid1,mid2+1,r2);
swapp(mid1+1,r1,l2,mid2);
{
int pp=0;
for(int i=l1+1;i<=r1;i++)
if(a[i]<a[i-1]){pp=1;break;}
for(int i=l2+1;i<=r2;i++)
if(a[i]<a[i-1]||pp){pp=1;break;}
if(!pp) res+=dfs(x+1,k+1);
}
swapp(mid1+1,r1,l2,mid2);
swapp(mid1+1,r1,mid2+1,r2);
{
int pp=0;
for(int i=l1+1;i<=r1;i++)
if(a[i]<a[i-1]){pp=1;break;}
for(int i=l2+1;i<=r2;i++)
if(a[i]<a[i-1]||pp){pp=1;break;}
if(!pp) res+=dfs(x+1,k+1);
}
swapp(mid1+1,r1,mid2+1,r2);
return res;
}
}
inline int read()
{
int s=0;char a=getchar();
while(a<'0'||a>'9')a=getchar();
while(a>='0'&&a<='9'){s=s*10+a-'0';a=getchar();}
return s;
}
signed main()
{
// freopen("in.txt","r",stdin); n=read();for(int i=1;i<=n;i++)m*=2;
for(int i=1;i<=m;i++)a[i]=read();
jc[0]=1;for(int i=1;i<=12;i++)jc[i]=jc[i-1]*i;
printf("%lld\n",dfs(0,0));
}

【SDOI2015】bzoj3990 排序的更多相关文章

  1. [SDOI2015][bzoj3990] 序列 [搜索]

    题面 传送门 思路 首先,这道题目有一个非常显然(但是我不会严格证明,只能意会一下)的结论:一个合法的操作序列中,任意两个操作是可以互换的 那么,这个结论加上本题极小的数据范围,为什么不搜索一下呢? ...

  2. BZOJ3990 排序(sort)

    排序(sort) 题目描述 小A有一个1~2N的排列A[1..2N],他希望将数组A从小到大排序.小A可以执行的操作有N种,每种操作最多可以执行一次.对于所有的i(1<=i<=N),第i种 ...

  3. 【LOJ】#2181. 「SDOI2015」排序

    题解 还以为是啥毒瘤题 然后是个搜索题 复杂度算起来挺大 然后跑起来就连0.1ms不到= = 就是从大到小进行每种操作,搜出来一种操作就乘上一个操作数的阶乘就行 如果现在进行的操作操作\(2^i\)那 ...

  4. BZOJ3990 排序

    题目:www.lydsy.com/JudgeOnline/problem.php?id=3990 这题很不错. 刚开始时无从下手,想了好多$O((2^n)log(2^n))$ 的idea,但是都不行. ...

  5. [BZOJ3990][SDOI2015]排序(DFS)

    3990: [SDOI2015]排序 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 902  Solved: 463[Submit][Status][ ...

  6. Bzoj3990 [SDOI2015]排序

    Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 651  Solved: 338 Description 小A有一个1-2^N的排列A[1..2^N], ...

  7. BZOJ3990:[SDOI2015]排序——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=3990 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作 ...

  8. [bzoj3990][SDOI2015]排序-搜索

    Brief Description 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的i(1<=i<= ...

  9. BZOJ3990 [SDOI2015]排序 【搜索】

    题目 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的i(1<=i<=N),第i中操作为将序列从左到 ...

随机推荐

  1. mysql 主从复制 配置

    mysql 的 默认配置文件在 /etc/my.cnf 1 修改主库 配置文件: 设置 服务id,并且开启二进制日志文件. server-id=1 log-bin=mysql-bin 2重启服务:se ...

  2. 学习Python笔记---操作列表

    1.for循环: 编写for循环时,对于用语存储列表中每个值的临时变量,可指定任何名称. 在for循环中,想包含多少行代码都可以,每个缩进的代码行都是循环的一部分,且将针对列表中的每个值都执行一次. ...

  3. IO多路复用,协程

    https://www.cnblogs.com/wangjun187197/p/9642429.html Python之路--协程/IO多路复用 I/O复用模型 此模型用到select和poll函数, ...

  4. oracle创建定时任务

    一.dmbs_job dbms_job涉及到的知识点 1.创建job: variable jobno number; dbms_job.submit(:jobno, —-job号  'your_pro ...

  5. leetcode 839 Similar String Groups

    题目 Two strings X and Y are similar if we can swap two letters (in different positions) of X, so that ...

  6. 第十章—DOM(二)——Element类型

    Element类型用于表现HTML和XML,提供了对元素标签名,子节点和特效的访问.Element节点具有以下特征: 要访问元素的标签名,可以使用nodeName属性,也可以使用tagName属性.这 ...

  7. jsp之jstl(展示所有商品、重写登录案例)

    jsp之jstl jstl: jsp标准的标签库语言,apache的,是用来替代java脚本 使用步骤: 1.导入jar包 (jstl.jar和standard.jar) 2.在页面上导入标签库 &l ...

  8. word之图表目录中点号位置提升3磅

  9. 50倍时空算力提升,阿里云RDS PostgreSQL GPU版本上线

    2019年3月19日,阿里云RDS PostgreSQL数据库GPU规格版本正式上线,开启了RDS异构计算并行加速之路.该版本在RDS(关系型数据库服务)的云基础设施层面首次完成了与阿里云异构计算产品 ...

  10. git reset三种模式

    reset三种模式区别和使用场景 区别: --hard:重置位置的同时,直接将 working Tree工作目录. index 暂存区及 repository 都重置成目标Reset节点的內容,所以效 ...