八皇後問題 遞歸實現 C語言 超詳細 思路 基礎

来源:https://www.cnblogs.com/cnnnnnn/archive/2018/03/04/8506883.html
-Advertisement-
Play Games

八皇後問題 :假設 將八個皇後放到國際象棋盤上,使其兩兩之間無法相互攻擊。共有幾種擺法? 基礎知識: 國際象棋里,棋盤為8X8格。 皇後每步可以沿直線、斜線 走任意格。 思路: 1.想把8個皇後放進去,肯定最終每行只有一個皇後,每列只有一個皇後。 2.設個二維數組chess [ i ] [ j ] ...


八皇後問題 :假設 將八個皇後放到國際象棋盤上,使其兩兩之間無法相互攻擊。共有幾種擺法?

基礎知識:

國際象棋里,棋盤為8X8格。

皇後每步可以沿直線、斜線 走任意格。

思路:

1.想把8個皇後放進去,肯定最終每行只有一個皇後,每列只有一個皇後。

2.設個二維數組chess [ i ] [ j ] 模擬棋盤,cas存放擺法。i j 是表示i行j列:

寫一個用於遞歸的函數,思路如下

3.從上往下一行行的放皇後,放下一行時從最左邊(第0列)放起,如果不能放就往右挪一格再試。註意判斷右邊有沒有越界出棋盤。

4.寫一個函數專門判斷當前位置能不能放,只需要判斷該位置的橫、豎、兩對角線,這四條線上有沒有其他皇後即可。命名為check。

5.如果把最後一行放完了,那就統計上這個擺法,cas++。擺完最後一行不能繼續判斷下一行了。

6.放完一種情況,還要探究其他情況,可以把現在放好的皇後“拿走”,然後再試探 之前沒試探過的棋盤格。

7.拿走皇後操作可以和不能放皇後的操作用同樣的代碼實現:

如果這個位置不能放,要把它置零,表示沒有皇後。

如果這位置能放,那就放皇後(置1)。等一種情況討論完,還得把它拿開,“拿開”也是置零的操作。

所以應該想辦法排列上述代碼,保證已經把擺出的情況記錄下來,之後執行“拿開皇後”代碼。

 

下麵是遞歸函數部分:

 

void queen(int i,int j){
if(j>=line){ //如果右側越界
return ;
}

if(check(i,j)==1){//如果能放
chess[i][j]=1;//放皇後
if(i==line-1){//如果是最後一行,記錄情況
cas++;
}
else{
queen(i+1,0);//不是最後一行就分析下一行
}
}
	//下麵這兩句是最精彩的
 chess[i][j]=0;//如果此位置不能放,就置空(0),判斷旁邊的格子。
//如果此位置能放,走到這裡就意味著上面的代碼全部執行了,把皇後拿走(置零),再討論其他情況,拿旁邊位置試探。
queen(i,j+1);

}

然後開始寫判斷函數check。需要判斷的是8個方向,把它看成4條直線考慮。對於所在的橫行,豎列,直接用for迴圈判斷。接下來考慮對角線(紅色)。

 

這樣可以看出來,要判斷的對角線是每個象限的平分線,每次 i ,j 的變化量是相等的,只是符號有差異。橫縱坐標變化量的範圍是-8~8,當對角線走到邊框時停止判斷。

為什麼是-8到8呢?因為咱們沒必要確定對角線的精確範圍,上圖是最理想的對角線,但是因為目標位置不同,對角線範圍也不同,每次計算兩端點是不可取的。

直接按最長對角線劃:


 

核心代碼:

for(k=-line;k<=line;k++){//兩對角線
if(i+k>=0&&i+k<line&&j+k>=0&&j+k<line)//從左上到右下對角線,如果在棋盤格裡
if(chess[i+k][j+k]==1) return 0;

if(i-k>=0&&i-k<line&&j+k>=0&&j+k<line)//從左下到右上對角線,如果在棋盤格裡
if(chess[i-k][j+k]==1) return 0;
}

 

這個判斷函數不重要,你寫的函數達到目的就行。

註意一點,此函數只能設計成判斷功能,不可以改變棋盤格的填充。

我剛開始覺得,當放置完上一個皇後,直接把這皇後所在的橫縱斜方向全部填充一個數字,列為‘‘禁地’’,下一皇後放置時只要看待放入位置是否是“禁地”即可。但這麼做是錯的,因為不是把棋盤格填好一次就ok了,擺好一次後還需要把最後放的棋子拿開,探討其他情況。如果劃分禁地後,拿走皇後還得把禁地複原了,很麻煩的說。。而且代碼量也不節省,就這麼判斷就行。

 

代碼塊的思路講完,下麵是完整代碼。運行結果:92.

你寫的程式也要保證這個結果。

 

#include<stdio.h>
#define line 8
void queen(int i,int j);
int check(int i,int j);

int queennumber=8;
int chess[line][line];
int cas=0;

int main(){
queen(0,0);
printf("%d\n",cas);
return 0;
}

void queen(int i,int j){
if(j>=line){
return ;
}

if(check(i,j)==1){//如果能放
chess[i][j]=1;//放皇後
if(i==line-1){//如果是最後一行,記錄情況
cas++;
}
else{
queen(i+1,0);//不是最後一行就分析下一行
}
}
chess[i][j]=0;//如果此位置不能放,就置空(0),判斷旁邊的格子。
//如果此位置能放,走到這裡就意味著上面的代碼全部執行了,把皇後拿走(置零),再討論其他情況,拿旁邊位置試探。
queen(i,j+1);
}


int check(int i,int j){
int k;

for(k=0;k<line;k++){
if(chess[i][k]==1) return 0;//0=不能放
}
for(k=0;k<line;k++){
if(chess[k][j]==1) return 0;
}
for(k=-line;k<=line;k++){//兩對角線
if(i+k>=0&&i+k<line&&j+k>=0&&j+k<line)//從左上到右下對角線
if(chess[i+k][j+k]==1) return 0;

if(i-k>=0&&i-k<line&&j+k>=0&&j+k<line)//從左下到右上對角線
if(chess[i-k][j+k]==1) return 0;
}
return 1;
}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • JsChart是什麼? JSChart能夠在網頁上生成圖標,常用於統計信息,十分好用的一個JS組件。 使用JsChart 一。導入jscharts.js 二。編寫jscharts.jsp測試頁面 1. 下載JScharts庫 從官網下載JScharts庫,我們使用的是壓縮包裡面的jscharts.j ...
  • 報錯 原因 缺少local.properties文件(SDK location) 解決 方法一:在android Studio中打開項目android目錄,會自動創建local.properties文件。 方法二:在android Studio根目錄中複製一份local.properties到rea ...
  • 一、MSA簡介 1.1、MSA是什麼 微服務架構MSA是Microservice Architecture的簡稱,它是一種架構模式,它提倡將單一應用程式劃分成一組小的服務,服務之間互相通訊、互相配合,為用戶提供最終價值。它與SOA之間的區別如下: SOA實現 微服務架構實現 企業級,自頂向下開展實施 ...
  • Constructor Constructor Constructor Constructor Constructor Constructor Constructor Constructor Constructor Rectangle Rectangle Rectangle Rectangle Re ...
  • http://python3-cookbook.readthedocs.io/zh_CN/latest/copyright.html ...
  • 交代背景 這篇帖子是為了提供我自己的July Novel站點的小說數據支撐。解決分散式部署爬蟲程式的繁瑣過程,由於本人對shell編程並不熟悉,故而先逐步記錄操作步驟,通過以下操作達到節省時間的方式。 三個前提: 1.首先是四台雲伺服器,全部安裝Cent OS 7.4, 四台伺服器中一臺主伺服器,三 ...
  • 數據類型為set。可以保證set內數據唯一。場景:生成訂單號,因為要求訂單號是絕對不能重覆的,所以資料庫中要設置為unique索引。但是其實可以通過redis,set來做每天的訂單集合。比如A客戶的訂單號201803041,B客戶併發了相同的訂單號,但是A客戶插入了set集合,B客戶插入就會返回0, ...
  • 常用來製作隊列,當然lpush+rpop也能做棧 #將RPUSH RPUSHX LPUSH LPUSHX一併介紹(具體介紹RPUSH和RPUSHX,因為其實就是插入的方向的區別) RPUSH key value [value ...] 向存於 key 的列表的尾部插入所有指定的值。如果 key 不存 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...