【问题描述】
给定一个 1~n 的排列 x,每次你可以将 x1~xi 翻转。你需要求出将序列变为
升序的最小操作次数。有多组数据。
【输入格式】
第一行一个整数 t 表示数据组数。
每组数据第一行一个整数 n,第二行 n 个整数 x1~xn。
【输出格式】
每组数据输出一行一个整数表示答案。
【样例输入输出】
sequence.in
1
8
8 6 1 3 2 4 5 7
sequence.out
7
【数据范围】
对于 100%的测试数据,t=5,n<=25。
对于测试点 1,2,n=5。
对于测试点 3,4,n=6。
对于测试点 5,6,n=7。
对于测试点 7,8,9,n=8。
对于测试点 10,n=9。
对于测试点 11,n=10。
对于测试点 i (12<=i<=25),n=i。
每次将 n 翻转到 x1 再翻转到 xn,可以得到一个不超过
2n-2 步的做法。由于步数不多,我们可以使用迭代加深搜索。
我们发现每次翻转只会改变一对相邻数对,因此对于一个状态求出相差>1 的相邻数对的数量,剩余步数一定大于这个值。加上这个剪枝就能通过本题。
时间复杂度 O(能过)
1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<algorithm> 5 #include<map> 6 using namespace std; 7 int a[26],ans,n,zyys,T; 8 int find() 9 {int i; 10 for (i=n;i>=1;i--,zyys++) 11 if (a[i]!=i) return i; 12 } 13 int count() 14 {int i; 15 int cnt=0; 16 for (i=n-1;i>=1;i--,zyys++) 17 if (abs(a[i]-a[i+1])!=1) cnt++; 18 return cnt; 19 } 20 bool dfs(int x) 21 {int i,j,flag; 22 int aa[26]={0}; 23 //if (zyys>90000000) return 1; 24 if (x>ans) return 0; 25 if (x+count()>ans) return 0; 26 flag=1; 27 for (i=n;i>=1;i--,zyys++) 28 if (a[i]!=i) 29 {flag=0;break;} 30 int r=i; 31 if (flag) 32 { 33 ans=x; 34 return 1; 35 } 36 memcpy(aa,a,sizeof(aa)); 37 zyys+=n; 38 for (i=1;i<=r;i++) 39 {zyys++; 40 for (j=1;j<=i;j++,zyys++) 41 a[j]=aa[i-j+1]; 42 if (dfs(x+1)) return 1; 43 for (j=1;j<=i;j++,zyys++) 44 a[j]=aa[j]; 45 } 46 return 0; 47 } 48 int main() 49 {int i,pos; 50 cin>>T; 51 while (T--) 52 { 53 cin>>n; 54 ans=n*2-2; 55 zyys=0; 56 for (i=1;i<=n;i++) 57 {zyys++; 58 scanf("%d",&a[i]); 59 if (a[i]==n) pos=i; 60 } 61 for (i=1;i<=2*n-2;i++) 62 {ans=i; 63 if (dfs(0)) break; 64 } 65 cout<<ans<<endl; 66 } 67 }