经典算法-----农夫过河问题(深度优先搜索)

目录

前言

农夫过河问题

1.问题描述

2.解决思路

位置编码

获取位置

判断是否安全 

 深度优先遍历(核心算法)

 3.完整代码


前言

        今天我们来解决一个有意思的问题,也就是农夫过河问题,可能这个问题我们小时候上学就听说过类似的问题,当时我们的解决方法就是一个一个列举,反复去找,但是在编程上对于这个问题的解决方法就是去通过回溯问题来找出全部的可能来解决这个问题,下面就一起来看看吧。

农夫过河问题

1.问题描述

一位农夫、一只狼、一只羊和一颗白菜都在河的一侧,他们想到河的另一侧。河中有一条船,一次只能载农夫和一个物体(狼或羊或白菜)。若农夫不在的时候,狼会吃掉羊,羊会吃掉白菜。问:该以什么样的方式才能将狼、羊和白菜完好的运到对岸?

在此之前我们先去通过自己思考一下怎么去过河,很简单:① 农夫把羊运到河岸1;② 农夫回来河岸0;③ 把白菜运往河岸1;④ 把羊运到河岸0;⑤ 把狼运到河岸1;⑥ 再回来河岸0;⑦ 把羊运往河岸1,最后完成过河。当然方法不止这一种,如果让你去写编程的话找出全部的方法,怎么写呢?

2.解决思路

位置编码

不妨设,当前其实岸为南岸即标记为0,目标岸北岸标记为1,然后分别通过一个二进制数来表示农夫、狼、羊、白菜的当前位置,比如1000(十进制为8)表示农夫当前位置在北岸,其他三个在南岸,以此类推0100(十进制为4)表示狼在北岸,0010(十进制为2)表示羊在北岸,0001(十进制为1)表示白菜在北岸,通过以上的规则我们可以明确的表示这4者的位置关系。

获取位置

        前面既然有了编码来表示位置关系,那么我假设当前4者的关系位置为 location,那如何获取到当前农夫、狼、羊、白菜的位置呢?

        很简单,只需要进行与运算就行了,比如location=1010,即表示农夫在北岸,狼在南岸,羊在北岸、白菜在南岸,那么我要知道农夫的位置只需要进行 location&8 的运算即可,得出的二进制数结果就是1000(十进制为8),即农夫在北岸.

//判断当前位置
int farmer(int loca) {//loca是表示当前4者的位置状态二进制数return ((loca & 8) == 8);
}
int wolf(int loca) {return ((loca & 4) ==4);
}
int sheep(int loca) {return ((loca & 2) ==2) ;
}
int cabbage(int loca) {return  ((loca & 1)==1);
}
判断是否安全 

既然有了位置,那就要去判断这种情况是否安全了,农夫不在狼和羊不可以在一起,农夫不在白菜和羊不可以在一起,

int isSafe(int loca) {int a, b, c, d;a = farmer(loca);b = wolf(loca);c = sheep(loca);d = cabbage(loca);if (a != c && c == d)//农夫不在白菜和羊不可以在一起,不安全return 0;else if (a != b && b == c)//农夫不在狼和羊不可以在一起,不安全return 0;elsereturn 1;
}
 深度优先遍历(核心算法)

        要想找到全部的运输方法那就需要去进行深度遍历,由于总共有16个状态,所以在此之前需要一个数组来储存表示当前的位置状态route[16],全部初始化为-1表示没有没有访问过这个运算过程,其中第一个状态也就是0000,最开始的时候4者都在南岸,那么route[0]=-2,表示最开始状态不需要去访问或者修改操作。

每次带一个物品过河,我们可以通过循环来实现二进制数的左移,从白菜开始向左移动,直到农夫,movers表示当前要进行移动的物体,这时候我们就需要算出如果这个物品移动之后,下一个状态nextlocation是否安全,如果满足条件的话,那就吧进行这一步移动操作,如果不安全那就换一个物体去移动,如果直到移动到农夫也不安全,那就进行回溯到上一步,从上一步重新开始移动下一个物体,以此类推,这就是一个深度遍历回溯的过程。

 

//深度遍历核心算法(回溯算法)
void process(int loca,int* route,int* count) {if (route[15] == -1) {//move表示要移动当前物体for (int move = 1; move <= 8; move <<=1) {//如果农夫与当前要移动的物体处于同一个岸的话if (((loca & 8)!=0) == ((loca & move)!=0)) {				int next_loca = loca ^ (8 | move);//获取下一个状态next_loca二进制数if (isSafe(next_loca) && route[next_loca] == -1) {//判断下一个状态是否安全,同时也没有访问过int next_route[16];for (int i = 0; i < 16; i++) {//把当前的路径复制一份,进入到下一步递归,以保证这一步的路径数据没被修改,进行回溯操作next_route[i] = route[i];}next_route[next_loca] = loca;//存储当前位置,进入到下一个process(next_loca, next_route,count);//回溯}}}}//如果route[15]储存了数据,那么都到达了对岸了,打印结果else {print(route, 15,count);puts("\n");}return;
}

 3.完整代码

#include<stdio.h>
#include<string.h>//北1  南0
//判断当前位置
int farmer(int loca) {return ((loca & 8) == 8);
}
int wolf(int loca) {return ((loca & 4) ==4);
}
int sheep(int loca) {return ((loca & 2) ==2) ;
}
int cabbage(int loca) {return  ((loca & 1)==1);
}int isSafe(int loca) {int a, b, c, d;a = farmer(loca);b = wolf(loca);c = sheep(loca);d = cabbage(loca);if (a != c && c == d)//农夫不在白菜和羊不可以在一起,不安全return 0;else if (a != b && b == c)//农夫不在狼和羊不可以在一起,不安全return 0;elsereturn 1;
}
//打印结果
void print(int* route,int status,int* count) {if (status == -2) {*count += 1;printf("第%d种方法:\n",*count);printf("start");return;}print(route, route[status],count);printf("->%d", status);
}//深度遍历核心算法(回溯算法)
void process(int loca,int* route,int* count) {if (route[15] == -1) {//move表示要移动当前物体for (int move = 1; move <= 8; move <<=1) {//如果农夫与当前要移动的物体处于同一个岸的话if (((loca & 8)!=0) == ((loca & move)!=0)) {				int next_loca = loca ^ (8 | move);//获取下一个状态next_loca二进制数if (isSafe(next_loca) && route[next_loca] == -1) {//判断下一个状态是否安全,同时也没有访问过int next_route[16];for (int i = 0; i < 16; i++) {//把当前的路径复制一份,进入到下一步递归,以保证这一步的路径数据没被修改,进行回溯操作next_route[i] = route[i];}next_route[next_loca] = loca;//存储当前位置,进入到下一个process(next_loca, next_route,count);//回溯}}}}//如果route[15]储存了数据,那么都到达了对岸了,打印结果else {print(route, 15,count);puts("\n");}return;
}int main() {int route[16];memset(route, -1, sizeof(route));route[0] = -2;int count = 0;//统计process(0,route,&count);
}

输出结果:

 以上就是今天的全部内容了,你们学会这个问题的解决方法吗?是不是很有意思呢?

分享一张壁纸: 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/151910.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Flink---10、处理函数(基本处理函数、按键分区处理函数、窗口处理函数、应用案例TopN、侧输出流)

星光下的赶路人star的个人主页 我的敌手就是我自己&#xff0c;我要他美好到能使我满意的程度 文章目录 1、处理函数1.1 基本处理函数&#xff08;ProcessFunction&#xff09;1.1.1 处理函数的功能和使用1.1.2 ProcessFunction解析1.1.3 处理函数的分类 1.2 按键分区处理函数&…

你的librosa和scikit-learn打架了吗?

被这个问题困扰好久&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 我的原来版本librosa0.7.1 和 scikit-learn1.3.1 一直拆了按&#xff0c;按…

【AI视野·今日Robot 机器人论文速览 第四十五期】Mon, 2 Oct 2023

AI视野今日CS.Robotics 机器人学论文速览 Mon, 2 Oct 2023 Totally 42 papers &#x1f449;上期速览✈更多精彩请移步主页 Interesting: &#x1f4da;PONG, Probabilistic Object Normals for Grasping 用于抓取的概率目标归一化&#xff0c;根据目标表面法向量获取的不确定…

抄写Linux源码(Day17:你的键盘是什么时候生效的?)

回忆我们需要做的事情&#xff1a; 为了支持 shell 程序的执行&#xff0c;我们需要提供&#xff1a; 1.缺页中断(不理解为什么要这个东西&#xff0c;只是闪客说需要&#xff0c;后边再说) 2.硬盘驱动、文件系统 (shell程序一开始是存放在磁盘里的&#xff0c;所以需要这两个东…

网络安全--安全认证、IPSEC技术

目录 1. 什么是数据认证&#xff0c;有什么作用&#xff0c;有哪些实现的技术手段&#xff1f; 2. 什么是身份认证&#xff0c;有什么作用&#xff0c;有哪些实现的技术手段&#xff1f; 3. 什么是VPN技术&#xff1f; 4. VPN技术有哪些分类&#xff1f; 5. IPSEC技术能够…

JavaScript-mooc(纯分享)

第一步下载软件 mooc_v1.3.2_windows_amd64.zip - 蓝奏云 解压后打开有这么多文件 用记事本的打开方式打开config的文件 第一个尖头改成你学校对应慕课英华网址 第二个箭头是你的账号 第三个箭头是你的密码 改好后点击文件保存 最后一步点击运行 {"global": {&qu…

一篇理解网络分层原理

一、网络分层的必要性。 如图是一个数据的传输过程&#xff0c;在这个途中会有很多的原因导致数据丢失&#xff0c;网络分层就要可以很大程度的避免这个现象。 网络分层的必要性体现在以下几个方面&#xff1a; 抽象复杂度&#xff1a;网络分层将网络功能按照不同的层次进行分…

基于Kylin的数据统计分析平台架构设计与实现

目录 1 前言 2 关键模块 2.1 数据仓库的搭建 2.2 ETL 2.3 Kylin数据分析系统 2.4 数据可视化系统 2.5 报表模块 3 最终成果 4 遇到问题 1 前言 这是在TP-LINK公司云平台部门做的一个项目&#xff0c;总体包括云上数据统计平台的架构设计和组件开发&#xff0c;在此只做…

C++ - C++11历史 - 统一列表初始化 - aotu - decltype - nullptr - C++11 之后 STL 的改变

目录 C的发展史了解 统一列表初始化 {}初始化 std::initializer_list 在 vector 的模拟实现当中加上 initializer_list 作为参数的构造函数 声明 aotu decltype nullptr C11 之后 STL 的一些变化 新容器 容器中的一些新方法 C的发展史了解 在2003年C标准委员会曾经提…

基于SSM的实验室考勤管理系统设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;采用JSP技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#x…

CCF CSP认证 历年题目自练Day25

题目 试题编号&#xff1a; 201403-3 试题名称&#xff1a; 命令行选项 时间限制&#xff1a; 1.0s 内存限制&#xff1a; 256.0MB 问题描述&#xff1a; 问题描述   请你写一个命令行分析程序,用以分析给定的命令行里包含哪些选项。每个命令行由若干个字符串组成,它们之间…

插入排序/折半插入排序

插入排序/折半插入排序 插入排序 插入排序(英语&#xff1a;Insertion Sort)是一种简单直观的排序算法。它的工作原理是通过构建有序序列&#xff0c;对于未排序数据&#xff0c;在已排序序列中从后向前扫描&#xff0c;找到相应位置并插入。插入排序在实现上&#xff0c;通常…

光伏储能直流系统MATLAB仿真(PV光伏阵列+Boost DCDC变换器+负载+双向DCDC变换器+锂离子电池系统)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

前端 | AjaxAxios模块

文章目录 1. Ajax1.1 Ajax介绍1.2 Ajax作用1.3 同步异步1.4 原生Ajax 2. Axios2.1 Axios下载2.2 Axios基本使用2.3 Axios方法 1. Ajax 1.1 Ajax介绍 Ajax: 全称&#xff08;Asynchronous JavaScript And XML&#xff09;&#xff0c;异步的JavaScript和XML。 1.2 Ajax作用 …

javaee SpringMVC中json的使用

jsp <%--Created by IntelliJ IDEA.User: 呆萌老师:QQ:2398779723Date: 2019/12/6Time: 15:55To change this template use File | Settings | File Templates. --%> <% page contentType"text/html;charsetUTF-8" language"java" %> <%St…

蓝桥杯每日一题2023.10.7

跑步锻炼 - 蓝桥云课 (lanqiao.cn) 题目描述 题目分析 简单枚举&#xff0c;对于2的情况特判即可 #include<bits/stdc.h> using namespace std; int num, ans, flag; int m[13] {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; bool is_ren(int n) {if((n %…

《C++ Primer》第5章 语句

参考资料&#xff1a; 《C Primer》第5版《C Primer 习题集》第5版 5.1 简单语句&#xff08;P154&#xff09; 在一个表达式的末尾加上 ; 就构成了表达式语句&#xff0c;其作用是执行表达式并丢弃结果。 空语句 由单独的 ; 构成的语句为空语句。空语句常用于语法上需要一…

【网络】路由器和交换机的区别

&#x1f341; 博主 "开着拖拉机回家"带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——&#x1f390;开着拖拉机回家_Linux,大数据运维-CSDN博客 &#x1f390;✨&#x1f341; &#x1fa81;&#x1f341; 希望本文能够给您带来一定的帮助&#x1…

6.MySql连接SqlYog

MySql连接SqlYog SqlYog和navicat均是数据可视化工具&#xff0c;熟悉其一即可 SqlYog下载安装 连接&#xff0c;密码和端口号一定要正确&#xff01;&#xff01;&#xff01; 2.保存到数据库 创建数据库&表 创建数据库 创建成功 创建表 点击保存 查看表数据的…

jira 浏览器插件在问题列表页快速编辑问题标题

jira-issueTable-quicker 这是一个可以帮助我们在问题表格页快速编辑问题的浏览器插件 github 地址 功能介绍 jira 不可否认是一个可以帮助有效提高工作效率的工具&#xff0c;但是我们在使用 jira 时使用问题表格可以让我们看到跟多的内容而不用关注细节&#xff0c;但是目…