基于C++的農(nóng)夫過河問題算法設計與實現(xiàn)方法
本文實例講述了基于C++的農(nóng)夫過河問題算法設計與實現(xiàn)方法。分享給大家供大家參考,具體如下:
問題描述:
一個農(nóng)夫帶著—只狼、一只羊和—棵白菜,身處河的南岸。他要把這些東西全部運到北岸。他面前只有一條小船,船只能容下他和—件物品,另外只有農(nóng)夫才能撐船。如果農(nóng)夫在場,則狼不能吃羊,羊不能吃白菜,否則狼會吃羊,羊會吃白菜,所以農(nóng)夫不能留下羊和白菜自己離開,也不能留下狼和羊自己離開,而狼不吃白菜。請求出農(nóng)夫?qū)⑺械臇|西運過河的方案。
實現(xiàn)上述求解的搜索過程可以采用兩種不同的策略:一種廣度優(yōu)先搜索,另一種深度優(yōu)先搜索。這里介紹在廣度優(yōu)先搜索方法中采用的數(shù)據(jù)結構設計。
程序源碼:
/*********************************************** * 農(nóng)夫過河問題(P64 隊列的應用) * 約定:用四位二進制數(shù)分別順序表示農(nóng)夫、狼、白菜和羊的狀態(tài) * 即:{dddd} <=> {Farmer, Wolf, Cabbage, Goat} 其中:d={0,1} * 說明:0表示在東岸 1表示在西岸,初始狀態(tài)為0000,終止狀態(tài)為1111 ************************************************/ #include<stdio.h> #include<stdlib.h> #define MAXSIZE 16 typedef int EntryType; typedef struct queue { EntryType data[MAXSIZE]; int front,rear; //front隊頭,rear隊尾 }SeqQueue, * SeqQueuePtr; // 創(chuàng)建空隊列 SeqQueuePtr create_sequeue(void) { SeqQueuePtr pque; pque = (SeqQueuePtr)malloc(sizeof(SeqQueue)); if(pque){ pque->front = 0; pque->rear = 0; } else{ printf("Error: malloc() failed, out of memory!\n"); } return(pque); } int is_queEmpty(SeqQueuePtr pque) { return( pque->front == pque->rear ); } int is_queFull(SeqQueuePtr pque) { return( (pque->rear+1)%MAXSIZE == pque->front); } // 入隊 int enqueue(SeqQueuePtr pque, EntryType x) { if(is_queFull(pque)){ printf("Queue Overflow Error: trying to add an element onto a full queue\n"); return 1; } else{ pque->data[pque->rear] = x; pque->rear = (pque->rear + 1) % MAXSIZE; return 0; } } // 隊首元素出隊(返回0表示出隊異常,出隊操作前隊列為空) int dequeue(SeqQueuePtr pque, EntryType * e) { if(is_queEmpty(pque)){ printf("Empty Queue.\n"); return 0; } else{ *e = pque->data[pque->front]; pque->front = (pque->front + 1) % MAXSIZE; return 1; } } int is_farmer_crossed(int state) { return ((state & 0x08) != 0); } int is_wolf_crossed(int state) { return ((state & 0x04) != 0); } int is_cabbage_crossed(int state) { return ((state & 0x02) != 0); } int is_goat_crossed(int state) { return ((state & 0x01) != 0); } // 若狀態(tài)相容(安全)則返回1,否則返回0 int is_safe(int state) { if((is_goat_crossed(state) == is_cabbage_crossed(state)) && (is_goat_crossed(state) != is_farmer_crossed(state))) // 羊菜同岸且農(nóng)夫不在場 return(0); if((is_goat_crossed(state) == is_wolf_crossed(state)) && (is_goat_crossed(state) != is_farmer_crossed(state))) // 狼羊同岸且農(nóng)夫不在場 return(0); return(1); } void river_crossing_problem() { int route[16]; // 記錄已經(jīng)考慮過的狀態(tài) int state; // 記錄當前時刻的狀態(tài)(狀態(tài)編號的二進制形式即狀態(tài)本身) int aftercross; // 記錄漁夫當前的選擇(渡河對象)會導致的結果狀態(tài) int passenger; // 臨時變量,用于表達農(nóng)夫的選擇(對應二進制位為1表示選中該乘客) int results[16]={0}; // 輸出結果 int counter, i; SeqQueuePtr states_que; // states_que = create_sequeue(); // 創(chuàng)建“狀態(tài)”隊列 enqueue(states_que,0x00); // 初始狀態(tài)0000入隊 for(int i = 0; i < 16; i++){ route[i] = -1; } //route[0] = 0; while(!is_queEmpty(states_que) && (route[15] == -1)) { if( !dequeue(states_que, &state) ){ printf("Error: dequeue() - the queue is empty\n"); } // 依次考慮農(nóng)夫可能的選擇:攜帶羊、白菜和狼,以及農(nóng)夫只身渡河 for( passenger = 1; passenger<= 8; passenger <<= 1 ) { // 由于農(nóng)夫總是在過河,隨農(nóng)夫過河的也只能是與農(nóng)夫同側的東西 if(((state & 0x08) != 0) == ((state & passenger) != 0)){ // 如果農(nóng)夫與當前乘客在河岸的同一側 aftercross = state^( 0x08|passenger ); // 渡河后的情況 if(is_safe(aftercross) && (route[aftercross] == -1)){ // 如果渡河后狀態(tài)安全,則將其狀態(tài)入隊 route[aftercross] = state; // 將當前狀態(tài)的索引記錄到路徑數(shù)組中(下標索引為后續(xù)狀態(tài)值) enqueue(states_que, aftercross); } } }//end for() }//end while() // 輸出過河策略:0表示在東岸 1表示在西岸,初始狀態(tài)為0000,終止狀態(tài)為1111 if(route[15] != -1) { //printf("The reverse path is:\n"); counter = 0; for(state = 15; state != 0; state = route[state]){ //printf("The state is: %d \n",state); results[counter] = state; counter++; //if(state == 0) return; } for(i = 0; i< counter; i++){ state= results[i]; aftercross = results[i+1]; if(state & 0x08){ printf("農(nóng)夫從東岸到西岸:"); } else{ printf("農(nóng)夫從西岸到東岸:"); } switch(state^aftercross ){ case 12: printf("把狼帶過河\n"); break; case 10: printf("把菜帶過河\n"); break; case 9: printf("把羊帶過河\n"); break; default: printf("什么也不帶\n"); break; } } } else{ printf("No solution for this problem.\n"); } } int main(void) { river_crossing_problem(); system("pause"); return 0; }
運行結果:
希望本文所述對大家C++程序設計有所幫助。
您可能感興趣的文章
- 04-02c語言的正則匹配函數(shù) c語言正則表達式函數(shù)庫
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對數(shù)怎么表達
- 04-02c語言沒有round函數(shù) round c語言
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 01-10c語言求1+2+...+n的解決方法
- 01-10求子數(shù)組最大和的解決方法詳解
- 01-10深入理解約瑟夫環(huán)的數(shù)學優(yōu)化方法
- 01-10深入二叉樹兩個結點的最低共同父結點的詳解
- 01-10數(shù)據(jù)結構課程設計- 解析最少換車次數(shù)的問題詳解
- 01-10c語言 跳臺階問題的解決方法


閱讀排行
本欄相關
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機閱讀
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10C#中split用法實例總結
- 01-10delphi制作wav文件的方法
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-10SublimeText編譯C開發(fā)環(huán)境設置
- 01-11ajax實現(xiàn)頁面的局部加載