【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中操作为将序列从左到 ...
随机推荐
- 洛谷P1002 过河卒 [2017年4月计划 动态规划15]
P1002 过河卒 题目描述 棋盘上A点有一个过河卒,需要走到目标B点.卒行走的规则:可以向下.或者向右.同时在棋盘上C点有一个对方的马,该马所在的点和所有跳跃一步可达的点称为对方马的控制点.因此称之 ...
- Vue表单验证插件的制作过程
一.表单验证模块的构成 任何表单验证模块都是由 配置――校验――报错――取值 这几部分构成的. 配置: 配置规则 和配置报错,以及优先级 校验: 有在 change 事件校验, 在点击提交按钮的时候校 ...
- Pyhton AES_cbc解密
最近很多朋友问我加密解密有没有啥好推荐的方式,一般对AES的加密解密方式直接用在线加密或者解密就行,我为了方便测试,将网址以python脚本的形式写了出来,很简单的东西,2分钟搞定,随手记录一下~~ ...
- Kubernetes Ingress 日志分析与监控的最佳实践
摘要: Ingress主要提供HTTP层(7层)路由功能,是目前K8s中HTTP/HTTPS服务的主流暴露方式.为简化广大用户对于Ingress日志分析与监控的门槛,阿里云容器服务和日志服务将Ingr ...
- Python之collection
1.计数器(counter) Counter是对字典类型的补充,用于追踪值的出现次数. ps:具备字典的所有功能 + 自己的功能 c = Counter('abcdeabcdabcaba') prin ...
- Spring_Aop基于配置文件
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.sp ...
- springboot security 安全
spring security几个概念 “认证”(Authentication) 是建立一个他声明的主体的过程(一个“主体”一般是指用户,设备或一些可以在你的应用程序中执行动作的其他系统) . “授权 ...
- node写简单的爬虫(二)
上次我们已经成功的爬取了网站上的图片,现在我们把爬取的图片存储到本地 首先引入request var request=require('request'); http.get(url, functio ...
- 洛谷 P3951 小凯的疑惑 找规律
目录 题面 题目链接 题目描述 输入输出格式 输入格式: 输出格式: 输入输出样例 输入样例: 输出样例: 说明 思路 证明 AC代码 include<bits/stdc++.h> 题面 ...
- bzoj1295 最长距离
Description windy有一块矩形土地,被分为 N*M 块 1*1 的小格子. 有的格子含有障碍物. 如果从格子A可以走到格子B,那么两个格子的距离就为两个格子中心的欧几里德距离. 如果从格 ...