91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

遞歸與分治算法練習

發布時間:2020-08-10 19:22:36 來源:ITPUB博客 閱讀:142 作者:ckxllf 欄目:編程語言

  最近剛學習算法設計與分析的課程,所用教材是清華大學出版社王曉東編著的《算法設計與分析》。一道關于遞歸與分治算法的練習題如下:

  剛拿到題目覺得這題目似乎和遞歸分治沒有什么關系,但是O(1)的空間復雜度,以及O(n)的時間復雜度度就限制了解決方法,也就是分治和遞歸。(使用python語言只需幾行,用切片即可完成,這里附上極其弱智的代碼)

  def exchange(a,k):

  a=a[k:]+a[0:k]#列表切片

  return a

  ls=[1,2,3,4,5,6,7]

  print(exchange(ls,4))

  現在我們來思考這個遞歸分治算法。

  開始前先說明一下變量含義:

  start:左邊子數組開始位置下標

  sep:分割位置下標(左邊子數組結束位置下標)

  end:右邊子數組結束位置下標

  首先,是最簡單的情況,相信大家一定能想到,如果兩個子數組長度相等,直接遍歷子數組的長度,寫上三行交換代碼就可以解決了。(在這就不給出圖例了,簡單腦補一下即可)

  接下來,就是剩余兩種情況:分別是左邊子數組長度>右邊子數組長度以及左邊子數組長度<右邊子數組長度。我的基本想法就是長度小的一邊可以直接交換到位,長度長的一邊分成兩部分,一部分就是長度短的子數組長度,另一部分就是剩余部分長度。即:

  

遞歸與分治算法練習

  長數組用和短數組相同長度的元素和短數組元素一一交換,長數組剩余元素不動。第一次交換完成后短數組已經直接到位,接下來處理剩余元素長度即可,從而問題規模縮小,使用分治遞歸可以解決。

  下面圖例都是以這個數組為例{1,2,3,4,5,6,7}(紅色字體表示已經到位的元素)

  圖一(start=0,sep=4,end=6):

  判斷是左邊大于右邊;長度為2的兩對交換。1,2和6,7互換位置;6,7到位。start前進2位(start=2),sep不變,end也不變。

  判斷是左邊大于右邊;1,2和6,7互換位置;6,7,1,2到位。start再前進2位(start=4),sep不變,end也不變。

  判斷是左邊小于右邊;5和3互換位置;6,7,1,2,3到位。start前進1位(start=5),sep增1(sep=5),end不變。

  判斷是左邊等于右邊;5和4直接交換位置,所有元素全部到位。

  圖二(start=0,sep=1,end=6):

  判斷是左邊小于右邊;長度為2的兩對交換。1,2和3,4互換位置;3,4到位。start前進2位(start=2),sep前進1位(sep=3),end也不變。

  判斷是左邊小于右邊;1,2和5,6互換位置;3,4,5,6到位。start再前進2位(start=4),sep前進2位(sep=5),end也不變。

  判斷是左邊大于右邊;5和3互換位置;6,7,1,2,3到位。start前進1位(start=5),sep增1(sep=5),end不變。

  判斷是左邊等于右邊;2和1直接交換位置,所有元素全部到位。

  接下來是代碼呈現:

  public static void exchange(int a[],int start,int sep,int end)

  { 鄭州婦科醫院 http://www.zyfuke.com/

  int t;

  // 左邊子數組長度 = 右邊子數組長度

  if(end-sep==sep-start+1)

  {

  for (int i = start; i <=sep; i++)

  {

  t=a[i];

  a[i]=a[i+sep-start+1];

  a[i+sep-start+1]=t;

  }

  }

  // 左邊子數組長度 > 右邊子數組長度

  if(end-sep

  {

  for(int i=end;i>=sep+1;i--)

  {

  t=a[i];

  a[i]=a[i-(sep-start+1)];

  a[i-(sep-start+1)]=t;

  }

  // start=start+end-sep;

  exchange(a, start+end-sep, sep, end);

  //遞歸調用exchange方法

  // exchange(a, start, sep, end);

  }

  // 左邊子數組長度 < 右邊子數組長度

  if(end-sep>sep-start+1)

  {

  for(int i=start;i<=sep;i++)

  {

  t=a[i];

  a[i]=a[i+sep-start+1];

  a[i+sep-start+1]=t;

  }

  // start=sep+1;

  // sep=sep+sep-start+1;

  exchange(a, sep+1, sep+sep-start+1, end);

  //遞歸調用exchange方法

  exchange(a, start, sep, end);

  }

  }

  左邊子數組長度 >右邊子數組長度:

  左右兩邊交換,中間不動,交換后左邊部分完成,右邊遞歸,*start前進短的子數組的長度個單位,*短的子數組長度=end-sep,所以有start=start+end-sep;sep不變,end也不變。

  左邊子數組長度 <右邊子數組長度:

  左邊中間交換,右邊不動,交換后左邊部分完成,右邊遞歸,start前進短的子數組的長度個單位,短的子數組長度=sep-start+1,所以有start=start+sep-start+1=sep+1。sep也前進短子數組長度個單位,sep=sep+sep-start+1;end不變。

  測試:

  int a[]= {10,2,8,3,5,4,7,1};

  ...

  exchange(a, 0,4,a.length-1);

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

泰来县| 宜丰县| 西宁市| 鹿邑县| 和静县| 罗定市| 邵阳市| 香格里拉县| 福建省| 饶河县| 武功县| 工布江达县| 长寿区| 惠安县| 石河子市| 梓潼县| 凉山| 上栗县| 安陆市| 长阳| 右玉县| 西充县| 龙陵县| 陇西县| 萝北县| 游戏| 巴楚县| 康马县| 高州市| 简阳市| 北海市| 庆元县| 乌鲁木齐市| 布尔津县| 罗甸县| 荣昌县| 抚远县| 洛浦县| 洱源县| 南丰县| 西昌市|