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

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

数据结构与算法-- 八皇后问题(多种实现方案)

發(fā)布時(shí)間:2023/12/4 编程问答 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 数据结构与算法-- 八皇后问题(多种实现方案) 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

八皇后問題解法一(排列篩選法)

  • 本篇我們承接上一篇中的思想,想到了一個(gè)經(jīng)典的算法題,八皇后問題:
  • 題目:在8*8的國(guó)際象棋上擺放8個(gè)皇后,使得其互相不能攻擊,即任意兩個(gè)換后不能在同一行,同一列,或者同一對(duì)角線上。如下圖中所示,就是一個(gè)符合預(yù)期的擺放方式,問總共有多少中擺放方式。

  • 上圖中的數(shù)字代表此處放置一個(gè)皇后,并且從上到下依次是0~7 總共8個(gè)。

  • 分析:

    • 由于8個(gè)皇后任意兩個(gè)不能處在同一行,那么肯定每個(gè)皇后占據(jù)一行。
    • 有上圖看,我們必然可以用數(shù)組來標(biāo)識(shí)myQueen[8] 數(shù)組,數(shù)組中i 個(gè)數(shù)字表示位于第i 行的皇后
    • 例如以上圖為案例可以用數(shù)組 myQueen[8] = {4,2,0,5,7,1,3,6} 來標(biāo)識(shí)這種擺放的可能,
      • 就是我們直接用數(shù)組下標(biāo)當(dāng)成皇后所在的列
      • 用數(shù)組的下標(biāo)對(duì)應(yīng)的值當(dāng)成皇后所在的行
    • 那么現(xiàn)在我們的思路就是求出 myQueen數(shù)組的全排列,接著在篩選出符合要求的
    • 符合要求的情況:
      • 我們依據(jù)以上思路得到的全排列,天然就不會(huì)再同一行,也不會(huì)在同一列
      • 只需要篩選左右對(duì)角線上是否有其他數(shù)據(jù),那么只需要對(duì)比下標(biāo) i 和 j的差值與 i和j 下標(biāo)對(duì)應(yīng)的value的差值是否相等
      • i - j == myQueen[i] -myQueen[j] || j-i == myQueen[i] - myQueen[j]
      • 以上判斷可以用具體案例來驗(yàn)證此處略。
  • 依據(jù)如上分析有如下代碼:

/*** 八皇后問題* @author liaojiamin* @Date:Created in 15:03 2021/5/26*/ public class EightQueen {private static final List<Integer[]> myQueen = new ArrayList<>();public static void main(String[] args) {Integer[] eight = {0,1,2,3,4,5,6,7};getEightQueenPerMutain(eight, 0);List<Integer[]> realQueen = checkQueen(myQueen);for (Integer[] integers : realQueen) {for (int i = 0; i < integers.length; i++) {System.out.print(integers[i]+",");}System.out.println();}}/*** 檢查所有可能的八皇后排列,篩選出符合條件的:* i-j == integer[i] - integer[j] || j-i = integer[i]-integer[j] 情況的都是統(tǒng)一對(duì)角的* */public static List<Integer[]> checkQueen(List<Integer[]> myQueen){List<Integer[]> realEightQueen = new ArrayList<>();for (Integer[] integers : myQueen) {boolean isCheck = true;for (int i = 0; i < integers.length; i++) {if(!isCheck){break;}for (int j = i+1; j<integers.length;j++){if((i-j) == (integers[i] - integers[j])|| (j-i) == (integers[i] - integers[j])){isCheck = false;break;}}}if(isCheck){realEightQueen.add(integers);}}return realEightQueen;}/*** 八皇后問題排列解決方案* */public static void getEightQueenPerMutain(Integer[] eight, Integer start){if(eight == null || eight.length <= 1 || start == eight.length-1){Integer[] newEight = new Integer[eight.length];for (int i = 0; i < eight.length; i++) {newEight[i] = eight[i];}myQueen.add(newEight);}for (int i = start; i <= eight.length-1; i++) {Integer temp = eight[start];eight[start] = eight[i];eight[i] = temp;getEightQueenPerMutain(eight, start+1);temp = eight[start];eight[start] = eight[i];eight[i] = temp;}} }

八皇后問題解法二(動(dòng)態(tài)規(guī)劃)

  • 接上文中,依然可以用數(shù)組 myQueen[8] = {4,2,0,5,7,1,3,6} 來標(biāo)識(shí)這種擺放的可能

  • 那么上文中將所有排列列舉出來后在對(duì)排列集合進(jìn)行篩選得到最終的八皇后集合

  • 類似的思路,我們用窮舉法將數(shù)組myQueen中包含0 ~ 7 的所有數(shù)字的可能一一列舉,并且實(shí)時(shí)篩選,這樣我們可以一個(gè)步驟直接得到想要的集合

  • 經(jīng)過如上分析有如下代碼:

/*** 八皇后問題* @author liaojiamin* @Date:Created in 15:03 2021/5/26*/ public class EightQueen {private static final List<Integer[]> myQueen = new ArrayList<>();public static void main(String[] args) {for (int i = 0; i < 8; i++) {Integer[] eight = new Integer[8];getEightQueen(eight, i);}for (Integer[] integers : myQueen) {for (int i = 0; i < integers.length; i++) {System.out.print(integers[i] + ",");}System.out.println();}System.out.println(myQueen.size());}/*** 窮舉+篩選* 直接獲取八皇后問題最終結(jié)果* */public static void getEightQueen(Integer[] queen, int start){for (int i = 0; i < 8; i++) {queen[start] = i;if(checkout(queen, start)){if(start == queen.length -1){Integer[] realQueen = new Integer[8];for (int i1 = 0; i1 < queen.length; i1++) {realQueen[i1] = queen[i1];}myQueen.add(realQueen);}else {getEightQueen(queen, start + 1);}}}}/** * 校驗(yàn)是否符合八皇后問題規(guī)則 */public static boolean checkout(Integer[] queen, Integer end){Set<Integer> checkRepeat = new HashSet<>();for (int i = 0; i <= end; i++) {if(checkRepeat.contains(queen[i])){return false;}checkRepeat.add(queen[i]);for(int j = i+1; j<= end; j++){if(queen[i] == null || queen[j] == null){return false;}if((i-j) == (queen[i]- queen[j]) || (j-i) == (queen[i] - queen[j])){return false;}}}return true;} }

八皇后問題解法三(回朔遞歸)

  • 依然可以用數(shù)組 myQueen[8] = {4,2,0,5,7,1,3,6} 來標(biāo)識(shí)這種擺放的可能
  • 首先將第一個(gè)皇后放入第一列位置
  • 接著將第二個(gè)皇后分別放入第二三四等之后的位置,每次分別去檢查放入的位置是否與之前的位置中放入的皇后是否在同一列,同一對(duì)角線
  • 接著依次對(duì)地三個(gè),第四個(gè),直到第八個(gè)皇后重復(fù)第二步驟,找出每個(gè)皇后所有合法的位置
  • 方法三 與方法二的實(shí)現(xiàn)方式非常類似,區(qū)別在于篩選,方法三種的篩選只需要對(duì)比最后一個(gè)元素與之前的元素是否沖突,在方法三中是每個(gè)皇后的位置依次檢查,當(dāng)進(jìn)行到第5個(gè)皇后的時(shí)候,能保證前4個(gè)皇后的位置都是合法的,無需在檢查之前的。
  • 具體實(shí)現(xiàn)方案如下,方法三實(shí)現(xiàn)方案更好理解。
/*** 八皇后問題** @author liaojiamin* @Date:Created in 15:03 2021/5/26*/ public class EightQueen {private static final List<Integer[]> myQueen = new ArrayList<>();public static void main(String[] args) {getEightQueenRetro();print();}public static void print() {for (Integer[] integers : myQueen) {for (int i = 0; i < integers.length; i++) {System.out.print(integers[i] + ",");}System.out.println();}System.out.println(myQueen.size());}/*** 回朔遞歸*/public static void getEightQueenRetro() {Integer[] eight = new Integer[8];check(eight, 0);}//遞歸回朔便利每一種可能public static void check(Integer[] eight, int n) {if (n == eight.length) {myQueen.add(Arrays.copyOf(eight, eight.length));return;}for (int i = 0; i < eight.length; i++) {eight[n] = i;if(judge(eight, n)){check(eight, n+1);}}}//篩選public static boolean judge(Integer[] eight, int n){for(int i=0;i<n;i++){if(eight[i] == eight[n] || Math.abs(n-i) == Math.abs(eight[n] - eight[i])){return false;}}return true;} }

總結(jié)

  • 以上兩個(gè)方法的整體思路類似,當(dāng)題目需要我們按照一定規(guī)則擺放若干個(gè)數(shù)字的時(shí)候,我們可以先求出這些數(shù)字的所有可能,然后在一一判斷每個(gè)組合是否滿足題目給定的要求。
  • 方法一利用上一篇中的排列的算法得出所有排列可能,接著篩選,時(shí)間復(fù)雜度O( n3 )
  • 方法二 利用遞歸窮舉方法得出所有數(shù)字的可能,并同時(shí)篩選,時(shí)間復(fù)雜度也是O(n3)
  • 兩種算法的時(shí)間復(fù)雜度都并非很優(yōu),如有更好的算法思想,求在評(píng)論指出

上一篇:數(shù)據(jù)結(jié)構(gòu)與算法–字符串的排列組合問題
下一篇:數(shù)據(jù)結(jié)構(gòu)與算法-- 數(shù)組中出現(xiàn)次數(shù)超過一半的數(shù)字(時(shí)間復(fù)雜度的討論)

總結(jié)

以上是生活随笔為你收集整理的数据结构与算法-- 八皇后问题(多种实现方案)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 成人性生交生交视频 | 国产免费看黄 | 亚洲国产精品国自产拍av | 成人午夜视频在线 | av激情网站 | 朝鲜黄色片| 日韩一区不卡 | 日本在线一级 | 最新av在线网址 | 免费看国产曰批40分钟粉红裤头 | 国产一区欧美一区 | 蜜臀av性久久久久蜜臀aⅴ | 午夜伦伦 | 日本国产一级片 | 伦理片一区二区三区 | 一区二区三区免费播放 | 岛国片在线免费观看 | 中文字幕第28页 | 九九热免费 | 制服av网 | 国产精品99久久久久久久女警 | 91精品国产综合久久精品图片 | 国产精品videos | www,五月天,com| 亚洲国产第一页 | 日本高清视频免费看 | 性饥渴的农村熟妇 | 在线毛片观看 | 男人影院在线观看 | 韩日av在线| 操三八男人的天堂 | 69亚洲乱人伦 | 日韩中文字幕2019 | 草草在线影院 | 91在线免费观看网站 | 日本裸体动漫 | 日日日日日日bbbbbb | 91国内揄拍国内精品对白 | 亚洲日本香蕉视频 | 国产干b| 成人动漫h在线观看 | 国产福利91精品 | 2025国产精品 | 天堂av免费在线 | 欧美在线小视频 | 中文字幕一区二区人妻痴汉电车 | 日本一区二区高清不卡 | 97久久久久 | 高中男男gay互囗交观看 | 91成人免费 | 国产精品日本一区二区在线播放 | 亚洲成熟毛多妇女av毛片 | 免费在线h| 欧美福利在线观看 | 亚洲污片| 色网站在线免费观看 | 无码精品人妻一区二区三区漫画 | 国产精品一区二区网站 | 麻豆国产av超爽剧情系列 | 亚洲高清视频免费观看 | 欧洲黄视频| 亚洲av无码乱码在线观看富二代 | 欧美日韩激情在线观看 | 三级小视频在线观看 | wwwww在线观看| 五月依人网 | www四虎影院 | 色视频国产 | jizzjizz美国| 亚洲少妇视频 | 国产精品资源在线观看 | 狠狠操很很干 | 国产成人免费网站 | 蜜桃99视频一区二区三区 | 国产精品久久久久久久免费观看 | 欧美黄色成人 | av 日韩 人妻 黑人 综合 无码 | 日本黄色xxxx| 99视频99 | 国产a国产片国产 | 麻豆免费在线观看 | 91在线无精精品一区二区 | 久久影视| 国产视频在线观看一区 | 国产色播av在线 | 亚洲天堂av中文字幕 | 蜜臀尤物一区二区三区直播 | 六月婷婷综合 | 国产日韩欧美二区 | 欧洲视频在线观看 | 国产成人精品一区二区三区四区 | 国产视频三区 | 成年男女免费视频 | 国产精品中文在线 | 国产精品精品国产 | 清草视频| 男人操女人网站 | 五月激情六月 | av无码精品一区二区三区宅噜噜 |