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

news2025/1/18 3:31:32

目录

前言

农夫过河问题

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;
	else
		return 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;
	else
		return 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.coloradmin.cn/o/1070294.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

分布式缓存-Redis集群

单点Redis的问题 数据丢失问题 Redis是内存存储&#xff0c;服务重启可能会丢失数据 并发能力问题 单节点Redis并发能力虽然不错&#xff0c;但也无法满足如618这样的高并发场景 故障恢复问题 如果Redis宕机&#xff0c;则服务不可用&#xff0c;需要一种自动的故障恢复手段…

机器学习|深度学习|重磅推出---全网最全Numpy简明教程(一)

本教程面向初学者的Numpy简明教程&#xff0c;学好Numpy才能在深度学习、机器学习、AI等领域进军&#xff0c;如果感觉此文不错&#xff0c;欢迎和博主交流探讨 文章目录 1、Numpy介绍2、创建ndarray数组2.1、np.array2.2、np.ones2.3、np.zeros2.4、np.full2.5、np.eye2.6、np…

ChatGPT私有数据结合有什么效果?它难吗?

ChatGPT的出现可谓是惊艳了全世界&#xff0c;ChatGPT的问答能力通过了图灵测试&#xff0c;使其回答问题的方式与人类几乎无法区分。大家不甘于只在官方的对话页面问答&#xff0c;想利用 GPT 模型的自然语言能力结合私有数据开拓更多的应用场景。 | ChatGPT私有数据结合特点 …

ARM:使用汇编完成三个灯流水亮灭

1.汇编源代码 .text .global _start _start: 设置GPIOF寄存器的时钟使能LDR R0,0X50000A28LDR R1,[R0]ORR R1,R1,#(0x1<<5)STR R1,[R0]设置GPIOE寄存器的时钟使能LDR R0,0X50000A28LDR R1,[R0] 从r0为起始地址的4字节数据取出放在R1ORR R1,R1,#(0x1<<4) 第4位设…

day24-JS进阶(构造函数,new实例化,原型对象,对象原型,原型继承,原型链)

目录 构造函数 深入对象 创建对象三种方式 构造函数 new实例化执行过程(important!) 实例成员&静态成员 实例对象&实例成员 静态成员 内置构造函数 基本包装类型 Object Object.keys(obj)返回所有键组成的字符串数组 Object.values(obj)返回所有值组成的字…

C# 替换字符串最后一个逗号为分号

使用场景&#xff0c;sql语句的insert into table(c1,c2,c3) values (v1,v2,v3),(v1,v2,v3),(v1,v2,v3), 为了提高执行效率&#xff0c;在一个insert into中执行时&#xff0c;在循环中拼接语句&#xff0c;最后一个逗号需要替换为分号才能执行。 public static string Replace…

采集软件在市场营销中的应用价值

随着互联网的发展&#xff0c;市场竞争愈发激烈&#xff0c;如何获取准确、全面的市场信息成为企业成功的关键。数据利器作为一款强大的市场营销助手软件&#xff0c;具备多项功能&#xff0c;帮助您实现精准营销&#xff0c;发现商机。 软件功能&#xff1a; 搜索引擎采集&…

docker:修改容器的共享内存大小

错误提示&#xff1a; RuntimeError: DataLoader worker (pid 83709) is killed by signal: Bus error. It is possible that dataloader’s workers are out of shared memory. Please try to raise your shared memory limit. 解决办法&#xff1a; 1&#xff1a;创建新容器…

docker虚拟网桥和业务网段冲突处理

ifconfig查看docker虚拟网桥ip地址 docker inspect --format{{.Name}} - {{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}} $(docker ps -aq)查询所有容器的ip 修改docker-compose networks networks xxx-network: driver: bridge ipam: c…

web应用常见的其他漏洞总结

1.暴力破解用户名和密码 admin:admin, test:test, weblogic:weblogic, root:passwd 2. 扫敏感目录及备份文件 以ation 为扩展名的文件&#xff0c;通过7kb和k8&#xff0c;破壳扫描&#xff0c;扫描出来一个Web.rar文件&#xff0c;可获取MSSQL SSA连接用户名密码,通过测试…

2.6 宽带接入技术

思维导图&#xff1a; 前言&#xff1a; 我的理解&#xff1a; 1. **早期互联网接入技术的局限性**&#xff1a; - 作者首先回顾了早期用户通过电话线和调制解调器连接到互联网服务提供商&#xff08;ISP&#xff09;的方式&#xff0c;指出这种方式的速度上限是56 kbit/…

顺序表的简单介绍

目录 前提须知&#xff1a; 数据结构&#xff1a; 什么是数据结构&#xff1f; 数据结构特点&#xff1a; 为什么需要数据结构&#xff1a; 顺序表&#xff1a; 线性表&#xff1a; 与数组区别&#xff1a; 静态顺序表与动态顺序表&#xff1a; 二者之间的区别&#x…

应用安全系列之四十:登录常见问题以及预防方法

对于所有系统而言,登录是一个必备的而且最重要的功能。随着系统越来越复杂,服务越来越多,为了方便用户使用系统的服务,SSO应运而生,SSO虽然方便了用户使用系统,也增加了风险。因为一旦登录出现问题,就很容易通过登录访问整个系统。可见,对于登录如果没有控制好,攻击者…

HALCON的基础运用案例:- 例1- 3D点云的分割

前言&#xff1a; 在这个例子里面展示了用HALCON的操作函数segment_object_model_3d&#xff0c;来把一个输入的2.5D的3D图像进行分割。这里因为图像是一组圆柱体&#xff0c;有运用了一个物体的判别操作函数&#xff1a;dev_display_fitting_results。然后&#xff0c;自动给…

Python 代码调试

from pdb import set_trace as stx 是一个Python代码中常用的调试技巧之一&#xff0c;它用于在代码中插入断点以进行调试。这行代码的作用是将Python标准库中的 pdb&#xff08;Python Debugger&#xff09;模块中的 set_trace 函数导入&#xff0c;并将其重命名为 stx&#x…

ArcMap:第二届全国大学生GIS技能大赛(广西师范学院)详解-上午题

目录 01 题目 1.1 第一小题 1.2 第二小题 1.3 第三小题 1.4 数据展示 02 思路和实操 2.1 第一问思路 2.2 第一问操作过程 2.2.1 地理配准 2.2.2 镶嵌 2.2.2.1 第一种镶嵌方法 2.2.2.2 第二种镶嵌方法 2.2.3 裁剪 2.2.4 DEM信息提取 2.2.5 分类 2.3 第二问思路 …

DependsOn注解失效问题排查

文章目录 前言一、现象描述1.1.背景描述1.2.第一次修改&#xff0c;使用DependsOn注解1.3.第二次修改&#xff0c;设置方法入参 二、看看源码2.1.Spring实例化的源码2.2.调试2.3.验证 总结 前言 最近几天遇到一个比较有意思的问题&#xff0c;发现Spring的DependsOn注解失效&a…

强化学习框环境 - robogym - 学习 - 4

强化学习环境 - robogym - 学习 - 4 文章目录 强化学习环境 - robogym - 学习 - 4项目地址为什么选择 robogym如何消去目标位置的阴影&#xff1f;如何让物体颜色变得正确&#xff1f; 项目地址 https://github.com/openai/robogym 为什么选择 robogym 自己的项目需要做一些机…

小白自学笔记—网络安全(黑客笔记)

1.网络安全是什么 网络安全可以基于攻击和防御视角来分类&#xff0c;我们经常听到的 “红队”、“渗透测试” 等就是研究攻击技术&#xff0c;而“蓝队”、“安全运营”、“安全运维”则研究防御技术。 2.网络安全市场 一、是市场需求量高&#xff1b; 二、则是发展相对成熟入…

29 WEB漏洞-CSRF及SSRF漏洞案例讲解

目录 CSRF漏洞解释&#xff0c;原理等CSRF漏洞检测&#xff0c;案例&#xff0c;防御等防御方案2、设置随机Token3、检验referer来源 SSRF漏洞会比csrf漏洞重要一些SSRF_PHP&#xff0c;JAVA漏洞代码协议运用演示案例:SSRF_漏洞代码结合某漏洞利用测试 如何查找ssrf漏洞 SSRF漏…