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认为两个操作序列不同,当且仅当操作个数不同,或者至少一个操作不同(种类不同或者操作位置不同).
Input
第一行,一个整数N
Output
一个整数表示答案
Sample Input
7 8 5 6 1 2 4 3
Sample Output
HINT
100%的数据, 1<=N<=12.
题解:
......搜索送命题
对于菜得不能再菜的垃圾博主,搜索真是这世界上最难的算法(dalao勿喷)。
考试时看到这题,觉得是道数学题或者是一道数据结构加数学题,结果看题解发现是一道搜索。
不会写搜索的垃圾博主当场吓尿。
但考试时输出阶乘qj测试点得了15pts,运气不错
咳咳...言归正传
其实首先要想到一个性质就是对于一个操作序列的顺序是不影响他的合法性的,也就是说每次多加一种操作对答案的贡献是阶乘的这也就是我没爆零的原因。
那么我们就只需要确定每个操作是不是要选就可以了。
我们从小到大dfs,dfs(now,x)表示选了now种方法,现在判断第x种方法。
显然递归入口dfs(0,1)。
然后我们考虑怎样剪枝。
实际上交换的过程就是把不合法的子序列进行交换得到合法的子序列。
然后我们就可以每次扫一遍,找出对于每次操作x,不合法的序列个数及不合法序列的开始位置以便交换。
如果有两个以上不合法的子序列那就完戏了,直接return。
如果没有不合法的子序列,那么就不需要进行这种操作,直接搜下一层。
如果有一个这种子序列,那么交换他的前一半和后一半,再判断是否合法,再进行搜索。
如果有两个这种子序列,那么判断四种情况,在进行搜索。
很多神犇只说四种情况,并没有说明白是哪四种情况,坑害了我这种懵逼的蒟蒻
其实也很好想就是判断前一段的前一段和后一段的前一段交换,前一段的后一段和后一段的前一段,前一段的前一段和后一段的后一段,前一段的后一段和后一段的前一段。
就似这四种情况辣。
另外还有要注意的是就是在写交换和判断两个过程时一定要处理好循环的边界问题,博主就因为这个被卡了好长时间。
(附上本人丑陋的代码)、
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<cmath> 5 #include<queue> 6 #include<vector> 7 #include<algorithm> 8 using namespace std; 9 int fac[15],power[15];int n;int num; 10 int a[1<<12|5]; 11 int res=0; 12 int qpower(int a,int b){ 13 int ans=1; 14 while(b){ 15 if(b&1) ans=ans*a; 16 b>>=1; 17 a=a*a; 18 } 19 return ans; 20 } 21 int check(int x,int k){ 22 for(int i=1;i<power[k];i++){//不要右边界 23 if(a[x+i]!=a[x+i-1]+1){ 24 return 1; 25 } 26 } 27 return 0; 28 } 29 void swap(int x,int y,int k){ 30 for(int i=0;i<power[k];i++){//左右边界都不要 31 int t; 32 t=a[x+i]; 33 a[x+i]=a[y+i]; 34 a[y+i]=t; 35 } 36 } 37 void dfs(int now,int x){ 38 //cout<<x<<" "<<now<<endl; 39 if(x==n+1){ 40 res+=fac[now]; 41 return ; 42 } 43 int pos1=0;int pos2=0;int cnt=0; 44 for(int i=1;i<=num;i+=power[x]){ 45 if(check(i,x)){ 46 cnt++; 47 if(cnt>=3) break; 48 if(!pos2){ 49 if(!pos1){ 50 pos1=i; 51 } 52 else pos2=i; 53 } 54 } 55 } 56 //cout<<cnt<<" "<<pos1<<" "<<pos2<<endl; 57 if(cnt>2) return ; 58 if(cnt==0){ 59 dfs(now,x+1); 60 } 61 if(cnt==1){ 62 swap(pos1,pos1+power[x-1],x-1); 63 dfs(now+1,x+1); 64 swap(pos1,pos1+power[x-1],x-1); 65 66 } 67 if(cnt==2){ 68 swap(pos1,pos2,x-1); 69 if(check(pos1,x)||check(pos2,x)){ 70 swap(pos1,pos2,x-1); 71 } 72 else{ 73 dfs(now+1,x+1); 74 swap(pos1,pos2,x-1); 75 } 76 swap(pos1,pos2+power[x-1],x-1); 77 if(check(pos1,x)||check(pos2,x)){ 78 swap(pos1,pos2+power[x-1],x-1); 79 } 80 else{ 81 dfs(now+1,x+1); 82 swap(pos1,pos2+power[x-1],x-1); 83 } 84 swap(pos1+power[x-1],pos2,x-1); 85 if(check(pos1,x)||check(pos2,x)){ 86 swap(pos1+power[x-1],pos2,x-1); 87 } 88 else{ 89 dfs(now+1,x+1); 90 swap(pos1+power[x-1],pos2,x-1); 91 } 92 swap(pos1+power[x-1],pos2+power[x-1],x-1); 93 if(check(pos1,x)||check(pos2,x)){ 94 swap(pos1+power[x-1],pos2+power[x-1],x-1); 95 } 96 else{ 97 dfs(now+1,x+1); 98 swap(pos1+power[x-1],pos2+power[x-1],x-1); 99 } 100 } 101 } 102 int main(){ 103 fac[0]=1;power[0]=1; 104 for(int i=1;i<=12;i++){ 105 fac[i]=fac[i-1]*i; 106 power[i]=power[i-1]<<1; 107 } 108 scanf("%d",&n); 109 num=qpower(2,n); 110 for(int i=1;i<=num;i++){ 111 scanf("%d",&a[i]); 112 } 113 dfs(0,1); 114 //for(int i=1;i<=15;i++) cout<<power[i]<<" "<<fac[i]<<endl; 115 //cout<<power[15]; 116 printf("%d",res); 117 }
一道看题解后完全自己码出来,调出来的代码,还是很高兴的。
搜索水平还是太弱了啊,考试时都不知道暴力怎么打,一定要提升搜索的水平啊