BZOJ 3990: [SDOI2015]排序(搜索+剪枝)
[SDOI2015]排序
Description
小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的i(1<=i<=N),第i中操作为将序列从左到右划分为2^{N-i+1}段,每段恰好包括2^{i-1}个数,然后整体交换其中两段.小A想知道可以将数组A从小到大排序的不同的操作序列有多少个,小A认为两个操作序列不同,当且仅当操作个数不同,或者至少一个操作不同(种类不同或者操作位置不同).
下面是一个操作事例:
N=3,A[1..8]=[3,6,1,2,7,8,5,4].
第一次操作,执行第3种操作,交换A[1..4]和A[5..8],交换后的A[1..8]为[7,8,5,4,3,6,1,2].
第二次操作,执行第1种操作,交换A[3]和A[5],交换后的A[1..8]为[7,8,3,4,5,6,1,2].
第三次操作,执行第2中操作,交换A[1..2]和A[7..8],交换后的A[1..8]为[1,2,3,4,5,6,7,8].
Input
第一行,一个整数N
第二行,2^N个整数,A[1..2^N]
Output
一个整数表示答案
Sample Input
3
7 8 5 6 1 2 4 3
Sample Output
6
HINT
100%的数据, 1<=N<=12.
Source
Round 1 感谢ZKY制作非官方数据
分析:
显然,对于每一种方案,我改变其操作顺序,仍然不影响结果,所以我们只需枚举每种操作做与不做即可,然后将排列数加入答案即可。
考虑进一步优化,对于第i种操作,将序列分为若干个长为2^(i-1)的小段,由于每种操作只能用一次,则当每个小段各自元素都连续且递增,就不需要这种操作了(相当于每2^(i-1)个为一组,每组都排好了序,要想是整体进一步排序,只能依靠下一步操作来排了),如果有一个不连续递增,则将这组分为两份后交换即可,如果有2组不满足,分四种情况讨论,最后如果超过2个,显然无法使它们有序,则对这种方案剪枝。
关键思想是由局部有序一直操作直至全局有序,当枚举第i种操作时,此时2^(i-2)长度的段必然是有序的(不合法情况已去除),只需考虑2^(i-1)长度的段即可。
program sort;
var
a:array[..]of longint;
n,i,m:longint; ans,tot:int64;
procedure swap(x1,y1,x2,y2:longint);
var i,t:longint;
begin
for i:= to y1-x1 do
begin t:=a[x1+i]; a[x1+i]:=a[x2+i]; a[x2+i]:=t; end;
end;
function cheak:boolean;
var i:longint;
begin
for i:= to m do if a[i]<>a[i-]+ then exit(false);
exit(true);
end;
procedure dfs(x:longint);
var i,j,t,v,s,k:longint; l:int64; w:array[..]of longint;
begin
if x>=n then
begin if cheak=false then exit;l:=;for i:= to tot do l:=l*i; inc(ans,l); exit; end;
t:= shl (x+); v:= shl x; s:=v; k:=;
for i:= to m div t do
begin
if (a[s]+<>a[s+])or(a[s] mod v<>) then begin inc(k); if k> then exit;w[k]:=s; end;
s:=s+t;
end;
if k= then dfs(x+);
inc(tot);
if k= then begin swap(w[]-v+,w[],w[]+,w[]+v); dfs(x+); swap(w[]-v+,w[],w[]+,w[]+v); end;
if k= then
begin
if (a[w[]]+=a[w[]+])and(a[w[]]+=a[w[]+]) then
begin
swap(w[]-v+,w[],w[]-v+,w[]); dfs(x+); swap(w[]-v+,w[],w[]-v+,w[]);
swap(w[]+,w[]+v,w[]+,w[]+v); dfs(x+); swap(w[]+,w[]+v,w[]+,w[]+v);
end;
if (a[w[]]+=a[w[]-v+])and(a[w[]+v]+=a[w[]+]) then
begin swap(w[]-v+,w[],w[]+,w[]+v);dfs(x+); swap(w[]-v+,w[],w[]+,w[]+v);end;
if (a[w[]+v]+=a[w[]+])and(a[w[]]+=a[w[]-v+]) then
begin swap(w[]+,w[]+v,w[]-v+,w[]);dfs(x+); swap(w[]+,w[]+v,w[]-v+,w[]); end;
end;
dec(tot);
end;
begin
readln(n);m:= shl n;
for i:= to m do read(a[i]);
ans:=; tot:=;
dfs();
writeln(ans);
end.
BZOJ 3990: [SDOI2015]排序(搜索+剪枝)的更多相关文章
- BZOJ 3990: [SDOI2015]排序 [搜索]
3990: [SDOI2015]排序 题意:\(2^n\)的一个排列,给你n种操作,第i种把每\(2^{i-1}\)个数看成一段,交换任意两段.问是这个序列有序的操作方案数,两个操作序列不同,当且仅当 ...
- BZOJ 3990 [SDOI2015]排序 ——搜索
[题目分析] 可以发现,操作的先后顺序是不影响结果的,那么答案就是n!的和. 可以从小的步骤开始搜索,使得每一个当前最小的块都是上升的数列,然后看看是否可行即可. 复杂度好像是4^n [代码](哪里写 ...
- BZOJ.3990.[SDOI2015]排序(DFS)
题目链接 操作序列的顺序显然是无关的,所以只需按特定顺序求出一个长度为\(l\)的操作序列,它对答案的贡献为\(l!\). 我们从小到大枚举所有选择.若当前为第\(i\)个,如果有一段长度为\(2^i ...
- BZOJ 3990 [SDOI2015]排序
题解: 首先很容易看出各个操作是互不影响的,即对于一个合法的操作序列,我们可以任意交换两个操作的位置而不影响合法性. 因此我们可以忽略操作先后的影响,只考虑这个操作是否会出现在操作序列中. 如果用2n ...
- [bzoj3990][SDOI2015]排序-搜索
Brief Description 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的i(1<=i<= ...
- 【搜索】BZOJ 3990: 【Sdoi 2015】排序
3990: [SDOI2015]排序 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 336 Solved: 164[Submit][Status][ ...
- [BZOJ3990][SDOI2015]排序(DFS)
3990: [SDOI2015]排序 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 902 Solved: 463[Submit][Status][ ...
- hdu 5887 搜索+剪枝
Herbs Gathering Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)T ...
- 搜索+剪枝——POJ 1011 Sticks
搜索+剪枝--POJ 1011 Sticks 博客分类: 算法 非常经典的搜索题目,第一次做还是暑假集训的时候,前天又把它翻了出来 本来是想找点手感的,不想在原先思路的基础上,竟把它做出来了而且还是0 ...
随机推荐
- System.FormatException: GUID 应包含带 4 个短划线的 32 位数(xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx)。解决办法
查一下数据库的UID数据是否格式正确,如: 错误格式1: {E056BB36-D824-4106-A9C3-D8D8B9ADC1C 错误格式2: E056BB36-D824-4106-A9C3-D8D ...
- 深入理解计算机系统_3e 第十章家庭作业 CS:APP3e chapter 10 homework
10.6 1.若成功打开"foo.txt": -->1.1若成功打开"baz.txt": 输出"4\n" -->1.2若未能成功 ...
- 5-15 笔记 jtopo使用
Jtopo的核心对象有6个,分别是Stage(舞台对象),Scene(场景对象),Node(节点对象),Link(连线对象),Container(容器对象),Effect.Animate(动画效果) ...
- C# 文件操作 常用的类
File------实用类,提供许多静态方法,用于移动.删除.和复制文件. Directory------实用类,提供许多静态方法,用于移动.删除和复制目录. Path------ 实用类,用于处理路 ...
- react的redux中间件
redux 的中间件(对dispatch功能的拦截和增强,一般不用自己写,有插件) dispath是一个方法,可以自行中间拦截更改: store.dispatch = function(action) ...
- 【思维题 最大权闭合子图】loj#6045. 「雅礼集训 2017 Day8」价
又是经典模型的好题目 题目描述 人类智慧之神 zhangzj 最近有点胖,所以要减肥,他买了 NN 种减肥药,发现每种减肥药使用了若干种药材,总共正好有 NN 种不同的药材. 经过他的人脑实验,他发现 ...
- ajax全局变量的使用
var username; $.ajax({ type:"post", url:"a.action", data: {}, dataType: 'text', ...
- 六、Linux 文件基本属性
Linux 文件基本属性 Linux系统是一种典型的多用户系统,不同的用户处于不同的地位,拥有不同的权限.为了保护系统的安全性,Linux系统对不同的用户访问同一文件(包括目录文件)的权限做了不同的规 ...
- 二十一、MySQL NULL 值处理
MySQL NULL 值处理 我们已经知道 MySQL 使用 SQL SELECT 命令及 WHERE 子句来读取数据表中的数据,但是当提供的查询条件字段为 NULL 时,该命令可能就无法正常工作. ...
- java设计模式2--工厂模式
1.工厂模式特点 可以工厂获取我们所需要的类.我们不需要知道工厂的内部是如何实现的,我们只需要告诉工厂我们需要哪个类,工厂就会自动返回我想要的类. 简单来说:工厂帮我们隐藏了复杂的逻辑处理过程,我们只 ...