一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - C/C++ - 基于C++的農夫過河問題算法設計與實現方法

基于C++的農夫過河問題算法設計與實現方法

2021-06-01 15:20尼奧普蘭 C/C++

這篇文章主要介紹了基于C++的農夫過河問題算法設計與實現方法,簡單描述了農夫過河問題,并結合實例形式詳細分析了基于C++實現農夫過河問題的相關算法實現步驟與操作技巧,需要的朋友可以參考下

本文實例講述了基于C++的農夫過河問題算法設計與實現方法。分享給大家供大家參考,具體如下:

問題描述:

一個農夫帶著—只狼、一只羊和—棵白菜,身處河的南岸。他要把這些東西全部運到北岸。他面前只有一條小船,船只能容下他和—件物品,另外只有農夫才能撐船。如果農夫在場,則狼不能吃羊,羊不能吃白菜,否則狼會吃羊,羊會吃白菜,所以農夫不能留下羊和白菜自己離開,也不能留下狼和羊自己離開,而狼不吃白菜。請求出農夫將所有的東西運過河的方案。

實現上述求解的搜索過程可以采用兩種不同的策略:一種廣度優先搜索,另一種深度優先搜索。這里介紹在廣度優先搜索方法中采用的數據結構設計。

程序源碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
/***********************************************
 * 農夫過河問題(P64 隊列的應用)
 * 約定:用四位二進制數分別順序表示農夫、狼、白菜和羊的狀態
 *   即:{dddd} <=> {Farmer, Wolf, Cabbage, Goat} 其中:d={0,1}
 * 說明:0表示在東岸 1表示在西岸,初始狀態為0000,終止狀態為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;
// 創建空隊列
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);
}
// 若狀態相容(安全)則返回1,否則返回0
int is_safe(int state)
{
  if((is_goat_crossed(state) == is_cabbage_crossed(state)) &&
    (is_goat_crossed(state) != is_farmer_crossed(state))) // 羊菜同岸且農夫不在場
    return(0);
  if((is_goat_crossed(state) == is_wolf_crossed(state)) &&
    (is_goat_crossed(state) != is_farmer_crossed(state))) // 狼羊同岸且農夫不在場
    return(0);
  return(1);
}
void river_crossing_problem()
{
  int route[16];      // 記錄已經考慮過的狀態
  int state;        // 記錄當前時刻的狀態(狀態編號的二進制形式即狀態本身)
  int aftercross;     // 記錄漁夫當前的選擇(渡河對象)會導致的結果狀態
  int passenger;      // 臨時變量,用于表達農夫的選擇(對應二進制位為1表示選中該乘客)
  int results[16]={0};   // 輸出結果
  int counter, i;
  SeqQueuePtr states_que; //
  states_que = create_sequeue(); // 創建“狀態”隊列
  enqueue(states_que,0x00);   // 初始狀態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");
    }
    // 依次考慮農夫可能的選擇:攜帶羊、白菜和狼,以及農夫只身渡河
    for( passenger = 1; passenger<= 8; passenger <<= 1 )
    {
      // 由于農夫總是在過河,隨農夫過河的也只能是與農夫同側的東西
      if(((state & 0x08) != 0) == ((state & passenger) != 0)){
        // 如果農夫與當前乘客在河岸的同一側
        aftercross = state^( 0x08|passenger ); // 渡河后的情況
        if(is_safe(aftercross) && (route[aftercross] == -1)){
          // 如果渡河后狀態安全,則將其狀態入隊
          route[aftercross] = state; // 將當前狀態的索引記錄到路徑數組中(下標索引為后續狀態值)
          enqueue(states_que, aftercross);
        }
      }
    }//end for()
  }//end while()
  // 輸出過河策略:0表示在東岸 1表示在西岸,初始狀態為0000,終止狀態為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("農夫從東岸到西岸:");
      }
      else{
        printf("農夫從西岸到東岸:");
      }
      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++的農夫過河問題算法設計與實現方法

希望本文所述對大家C++程序設計有所幫助。

原文鏈接:http://blog.csdn.net/u011889952/article/details/44805069

延伸 · 閱讀

精彩推薦
  • C/C++學習C++編程的必備軟件

    學習C++編程的必備軟件

    本文給大家分享的是作者在學習使用C++進行編程的時候所用到的一些常用的軟件,這里推薦給大家...

    謝恩銘10102021-05-08
  • C/C++c++ 單線程實現同時監聽多個端口

    c++ 單線程實現同時監聽多個端口

    這篇文章主要介紹了c++ 單線程實現同時監聽多個端口的方法,幫助大家更好的理解和學習使用c++,感興趣的朋友可以了解下...

    源之緣11542021-10-27
  • C/C++C語言中炫酷的文件操作實例詳解

    C語言中炫酷的文件操作實例詳解

    內存中的數據都是暫時的,當程序結束時,它們都將丟失,為了永久性的保存大量的數據,C語言提供了對文件的操作,這篇文章主要給大家介紹了關于C語言中文件...

    針眼_6702022-01-24
  • C/C++C語言實現電腦關機程序

    C語言實現電腦關機程序

    這篇文章主要為大家詳細介紹了C語言實現電腦關機程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下...

    xiaocaidayong8482021-08-20
  • C/C++C/C++經典實例之模擬計算器示例代碼

    C/C++經典實例之模擬計算器示例代碼

    最近在看到的一個需求,本以為比較簡單,但花了不少時間,所以下面這篇文章主要給大家介紹了關于C/C++經典實例之模擬計算器的相關資料,文中通過示...

    jia150610152021-06-07
  • C/C++深入理解goto語句的替代實現方式分析

    深入理解goto語句的替代實現方式分析

    本篇文章是對goto語句的替代實現方式進行了詳細的分析介紹,需要的朋友參考下...

    C語言教程網7342020-12-03
  • C/C++詳解c語言中的 strcpy和strncpy字符串函數使用

    詳解c語言中的 strcpy和strncpy字符串函數使用

    strcpy 和strcnpy函數是字符串復制函數。接下來通過本文給大家介紹c語言中的strcpy和strncpy字符串函數使用,感興趣的朋友跟隨小編要求看看吧...

    spring-go5642021-07-02
  • C/C++C++之重載 重定義與重寫用法詳解

    C++之重載 重定義與重寫用法詳解

    這篇文章主要介紹了C++之重載 重定義與重寫用法詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下...

    青山的青6062022-01-04
主站蜘蛛池模板: 欧亚尺码专线欧洲s码wmy | 日本高清动作片www欧美 | 满溢游泳池免费土豪全集下拉版 | 精品国产乱码久久久久久免费 | 亚洲欧美日韩国产一区二区精品 | 我和子伦系列小说 | 精品久久久久久亚洲精品 | 久久丫线这里只精品 | 亚洲第一网色综合久久 | 日本中文字幕一区二区有码在线 | 欧美一级专区免费大片俄罗斯 | 国产一二三区视频 | 色偷偷91久久综合噜噜噜 | 美女任你模 | 国产福利一区二区三区 | 亚洲第一综合天堂另类专 | 亚洲国产第一 | 乌克兰少妇大胆大BBW | 国产欧美一区二区三区免费看 | a天堂视频 | 爱情岛永久成人免费网站 | 1024亚洲天堂 | 91九色最新地址 | a毛片免费全部在线播放毛 a级在线看 | 国产精品综合在线 | 狠狠综合网 | 免费的强动漫人物 | 成人久久伊人精品伊人 | 欧美日本一区视频免费 | 久久亚洲精品专区蓝色区 | 国产清纯91天堂在线观看 | 国产精品视频一区二区三区w | 国产精品麻豆 | 视频在线观看大片 | 火影小南被爆羞羞网站进入 | yellow视频在线观看 | 1313午夜精品久久午夜片 | 亚洲欧洲日产国码无码av | 国产午夜免费不卡精品理论片 | 狠狠色成人综合 | 精品一区heyzo在线播放 |