日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

jzoj4049-排序【搜索】

發布時間:2023/12/3 编程问答 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 jzoj4049-排序【搜索】 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

正題

題目鏈接:https://jzoj.net/senior/#contest/show/3017/0


題目大意

長度為2n2^n2n的序列,nnn個操作,第iii個可以將序列劃分為2i2^i2i段后交換其中兩段,每個操作只能用一次,求有多少種操作可以使得序列有小到大。


解題思路

每個操作只能一次操作順序不會影響結果,所以可以假設從小到大操作,做到第iii個操作時,每個可以被交換的段都必須是已經交換好的。

那么假設現在交換的段長度xxx,那么每段2x2x2x的最多只有兩個不按順序,否則無解,所以我們搜索一下就好了

時間復雜度O(2nn)O(2^nn)O(2nn)


codecodecode

#include<cstdio> #include<cstring> #include<algorithm> #define ll long long using namespace std; const ll N=14; ll n,l,ans,a[1<<N],fac[N]; void dfs(ll dep,ll f){if(dep>n){ans+=fac[f];return;}ll len=1<<dep,h=len/2,z=0,w[4];for(ll i=1;i<=l;i+=len){if(a[i+h]!=a[i]+h) w[++z]=i;if(z>2){return;}}if(!z)dfs(dep+1,f);if(z==1){if(a[w[1]]==a[w[1]+h]+h){for(ll i=w[1];i<w[1]+h;i++)swap(a[i],a[i+h]);dfs(dep+1,f+1);for(ll i=w[1];i<w[1]+h;i++)swap(a[i],a[i+h]);}}if(z==2){if(a[w[1]]+h==a[w[2]]&&a[w[1]+h]+h==a[w[2]+h]){for(ll i=0;i<h;i++)swap(a[w[1]+h+i],a[w[2]+i]);dfs(dep+1,f+1);for(ll i=0;i<h;i++)swap(a[w[1]+h+i],a[w[2]+i]);}if(a[w[2]]+h==a[w[1]+h]&&a[w[1]]+h==a[w[2]+h]){for(ll i=0;i<h;i++)swap(a[w[1]+i],a[w[2]+i]);dfs(dep+1,f+1);for(ll i=0;i<h;i++)swap(a[w[1]+i],a[w[2]+i]);}if(a[w[1]]+h==a[w[2]+h]&&a[w[2]]+h==a[w[1]+h]){for(ll i=0;i<h;i++)swap(a[w[1]+h+i],a[w[2]+h+i]);dfs(dep+1,f+1);for(ll i=0;i<h;i++)swap(a[w[1]+h+i],a[w[2]+h+i]);}if(a[w[2]+h]+h==a[w[1]+h]&&a[w[2]]+h==a[w[1]]){for(ll i=0;i<h;i++)swap(a[w[1]+i],a[w[2]+h+i]);dfs(dep+1,f+1);for(ll i=0;i<h;i++)swap(a[w[1]+i],a[w[2]+h+i]);}} } int main() {scanf("%lld",&n);l=1<<n;fac[0]=1;for(ll i=1;i<=n;i++)fac[i]=fac[i-1]*i;for(ll i=1;i<=l;i++)scanf("%lld",&a[i]);dfs(1,0);printf("%lld",ans); }

總結

以上是生活随笔為你收集整理的jzoj4049-排序【搜索】的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。