【SDOI2015】bzoj3990 排序
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 排序的更多相关文章
- [SDOI2015][bzoj3990] 序列 [搜索]
题面 传送门 思路 首先,这道题目有一个非常显然(但是我不会严格证明,只能意会一下)的结论:一个合法的操作序列中,任意两个操作是可以互换的 那么,这个结论加上本题极小的数据范围,为什么不搜索一下呢? ...
- BZOJ3990 排序(sort)
排序(sort) 题目描述 小A有一个1~2N的排列A[1..2N],他希望将数组A从小到大排序.小A可以执行的操作有N种,每种操作最多可以执行一次.对于所有的i(1<=i<=N),第i种 ...
- 【LOJ】#2181. 「SDOI2015」排序
题解 还以为是啥毒瘤题 然后是个搜索题 复杂度算起来挺大 然后跑起来就连0.1ms不到= = 就是从大到小进行每种操作,搜出来一种操作就乘上一个操作数的阶乘就行 如果现在进行的操作操作\(2^i\)那 ...
- BZOJ3990 排序
题目:www.lydsy.com/JudgeOnline/problem.php?id=3990 这题很不错. 刚开始时无从下手,想了好多$O((2^n)log(2^n))$ 的idea,但是都不行. ...
- [BZOJ3990][SDOI2015]排序(DFS)
3990: [SDOI2015]排序 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 902 Solved: 463[Submit][Status][ ...
- Bzoj3990 [SDOI2015]排序
Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 651 Solved: 338 Description 小A有一个1-2^N的排列A[1..2^N], ...
- BZOJ3990:[SDOI2015]排序——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=3990 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作 ...
- [bzoj3990][SDOI2015]排序-搜索
Brief Description 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的i(1<=i<= ...
- BZOJ3990 [SDOI2015]排序 【搜索】
题目 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的i(1<=i<=N),第i中操作为将序列从左到 ...
随机推荐
- org.hibernate.service.UnknownServiceException: Unknown service requested [org.hibernate.engine.jdbc.connections.spi.ConnectionProvider]
蜜汁问题? 完全不知道怎么解决啊
- MySQLDemo2
-- 查询所有数据库 show databases -- 删除数据库 drop database a -- use `数据库名称`; 表示使用此数据库 use mybatis -- 查看表结构 sho ...
- Leetcode643.Maximum Average Subarray I子数组的最大平均数1
给定 n 个整数,找出平均数最大且长度为 k 的连续子数组,并输出该最大平均数. 示例 1: 输入: [1,12,-5,-6,50,3], k = 4 输出: 12.75 解释: 最大平均数 (12- ...
- MaxCompute问答整理之7月
本文是基于本人对MaxCompute产品的学习进度,再结合开发者社区里面的一些问题,进而整理成文.希望对大家有所帮助. 问题一.DataWorks V2.0简单模式和标准模式的区别?公司数仓的数据上云 ...
- 获取电脑名和Ip
private string GetHostNameAndIP( bool isv4Orv6) { string HostName = Dns.GetHostN ...
- NLTK的探索
import nltk import random from nltk.corpus import movie_reviews documents = [(list(movie_reviews.wor ...
- vue常用操作及学习笔记(路由跳转及路由传参篇)
路由跳转 - 超链接方式跳转 html: <div id="app"> <h1>Hello App!</h1> <p> <!- ...
- VC程序异常中断的原因
自己编写的VC程序,编译调试通过,运行良好,然后关闭运行界面,就弹出一个“已经触发一个中断”的提示,然后断点就停在了下面这个中断处. _CRTIMP void _cdecl _CrtDbgBreak( ...
- 【Leetcode 堆、快速选择、Top-K问题 BFPRT】数组中的第K个最大元素(215)
这道题很强大,引出了很多知识点 题目 在未排序的数组中找到第 k 个最大的元素.请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素. 示例 1: 输入: [3,2,1,5 ...
- SAS信用评分之番外篇异常值的识别
SAS信用评分之番外篇异常值的识别 今天想分享给大家的是我早期建模的时候一个识别异常值的办法,也许你在"信用风险评分卡研究"看过,但是代码只能识别一个变量,我将这个代码作了改良,但 ...