列表

详情


NC20374. [SDOI2015]排序

描述

小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].

输入描述

第一行,一个整数N
第二行,2^N个整数,A[1..2^N]

输出描述

一个整数表示答案

示例1

输入:

3
  7 8 5 6 1 2 4 3

输出:

6

原站题解

上次编辑到这里,代码来自缓存 点击恢复默认模板

C++(g++ 7.5.0) 解法, 执行用时: 3ms, 内存消耗: 480K, 提交时间: 2022-12-30 10:18:33

#include <cstdio>
#include <cstdlib>
#include <iostream>
#include <algorithm>
using namespace std;
int num[5010],fac[20],ans=0;
int n;
  
bool check(int s,int pos)
{
  s=((s-1)<<pos)+1;
  int last=num[s];
  for(int i=2;i<=(1<<pos);i++)
  {
    if (num[s+i-1]!=last+1) return 0;
    else last=num[s+i-1];
  }
  return 1;
}
  
void Swap(int s1,int s2,int pos)
{
  for(int i=1;i<=(1<<pos);i++)
    swap(num[s1+i-1],num[s2+i-1]);
}
  
void dfs(int pos,int k)
{
  if (pos==n+1) {ans+=fac[k];return;}
  int p1=0,p2=0;
  for(int i=1;i<=(1<<(n-pos));i++)
  {
    if (!check(i,pos))
    {
      if (!p1) p1=i;
      else if (!p2) p2=i;
      else return;
    }
  }
  if (!p1&&!p2) dfs(pos+1,k);
  else if (p1&&!p2)
  {
    Swap(((p1-1)<<pos)+1,((p1-1)<<pos)+(1<<(pos-1))+1,pos-1);
    if (check(p1,pos)) dfs(pos+1,k+1);
    Swap(((p1-1)<<pos)+1,((p1-1)<<pos)+(1<<(pos-1))+1,pos-1);
  }
  else
  {
    for(int a=0;a<=1;a++)
      for(int b=0;b<=1;b++)
      {
        int s1,s2;
        if (!a) s1=((p1-1)<<pos)+1;
        else s1=((p1-1)<<pos)+(1<<(pos-1))+1;
        if (!b) s2=((p2-1)<<pos)+1;
        else s2=((p2-1)<<pos)+(1<<(pos-1))+1;
        Swap(s1,s2,pos-1);
        if (check(p1,pos)&&check(p2,pos))
          dfs(pos+1,k+1);
        Swap(s1,s2,pos-1);
      }
  }
}
  
int main()
{
  scanf("%d",&n);
  fac[0]=1;
  for(int i=1;i<=n;i++) fac[i]=fac[i-1]*i;
  for(int i=1;i<=(1<<n);i++)
    scanf("%d",&num[i]);
  dfs(1,0);
   
  printf("%d",ans);
   
  return 0;
}

C++11(clang++ 3.9) 解法, 执行用时: 4ms, 内存消耗: 504K, 提交时间: 2019-03-09 14:39:19

#include <cstdio>
#include <cstdlib>
#include <iostream>
#include <algorithm>
using namespace std;
int num[5010],fac[20],ans=0;
int n;
 
bool check(int s,int pos)
{
  s=((s-1)<<pos)+1;
  int last=num[s];
  for(int i=2;i<=(1<<pos);i++)
  {
    if (num[s+i-1]!=last+1) return 0;
    else last=num[s+i-1];
  }
  return 1;
}
 
void Swap(int s1,int s2,int pos)
{
  for(int i=1;i<=(1<<pos);i++)
    swap(num[s1+i-1],num[s2+i-1]);
}
 
void dfs(int pos,int k)
{
  if (pos==n+1) {ans+=fac[k];return;}
  int p1=0,p2=0;
  for(int i=1;i<=(1<<(n-pos));i++)
  {
	if (!check(i,pos))
	{
	  if (!p1) p1=i;
	  else if (!p2) p2=i;
	  else return;
	}
  }
  if (!p1&&!p2) dfs(pos+1,k);
  else if (p1&&!p2)
  {
    Swap(((p1-1)<<pos)+1,((p1-1)<<pos)+(1<<(pos-1))+1,pos-1);
	if (check(p1,pos)) dfs(pos+1,k+1);
	Swap(((p1-1)<<pos)+1,((p1-1)<<pos)+(1<<(pos-1))+1,pos-1);
  }
  else
  {
    for(int a=0;a<=1;a++)
	  for(int b=0;b<=1;b++)
	  {
		int s1,s2;
	    if (!a) s1=((p1-1)<<pos)+1;
		else s1=((p1-1)<<pos)+(1<<(pos-1))+1;
		if (!b) s2=((p2-1)<<pos)+1;
		else s2=((p2-1)<<pos)+(1<<(pos-1))+1;
		Swap(s1,s2,pos-1);
		if (check(p1,pos)&&check(p2,pos))
		  dfs(pos+1,k+1);
		Swap(s1,s2,pos-1);
	  }
  }
}
 
int main()
{
  scanf("%d",&n);
  fac[0]=1;
  for(int i=1;i<=n;i++) fac[i]=fac[i-1]*i;
  for(int i=1;i<=(1<<n);i++)
    scanf("%d",&num[i]);
  dfs(1,0);
  
  printf("%d",ans);
  
  return 0;
}

上一题