4.7 蚂蚁爬杆问题

2024-05-28 15:58
文章标签 问题 4.7 蚂蚁 爬杆

本文主要是介绍4.7 蚂蚁爬杆问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 前言

本文的一些图片, 资料 截取自编程之美

2. 问题描述

这里写图片描述

3. 问题分析

根据问题, 绘制一张状态图 :
这里写图片描述

解法一 : 模拟每一只蚂蚁的运动, 并以 各个蚂蚁的单位移动量需要的时间的最小公倍数 为周期检测所有的蚂蚁的情况, 检测碰撞, 统计每一只蚂蚁离开竹竿的时间, 最后的结果中找出最大, 最小值即为所求

解法二 : 将两只蚂蚁的碰撞 假想为两只蚂蚁的擦肩而过
这里写图片描述

那么, 这样一看起来, 所有的蚂蚁就”互不影响”了, 求出蚂蚁离开竹竿的最长, 最短时间, 就可以遍历一次蚂蚁, 获取他们离开的时间, 然后获取最大值, 最小值即可


这里 我的思路是找出最左边的向右的蚂蚁 和最右边的向左的蚂蚁, 然后找出二者的爬出竹竿的时间的较大者即为蚂蚁爬出竹竿的最长的时间, 爬出竹竿的最小时间类似

后来, 闲来无事的时候, 思考了一下后面的部分思考题
问题1. 试问每只蚂蚁爬出竹竿需要的时间分别是多少
问题2. 整个流程中, 所有的蚂蚁共碰撞了多少次

对于问题一 : 思路是来自这篇播客(或者转载这里, 具体的我也忘了) : http://lam8da.sinaapp.com/?p=11
所以, 我在Ant类中, 添加了一个dstId 用来表示在当前时间点之前, 碰撞的上一个蚂蚁的id, 出杆的时候, dstId对应的蚂蚁无障碍出杆需要的时间, 即为当前蚂蚁出杆需要的时间

这里, 我模拟运动过程的思路是 : 先移除竹竿最左边的向左走的蚂蚁(们), 向右边的向右走的蚂蚁(们) [之后最左边的蚂蚁必然向右, 最右边的蚂蚁必然向左] , 然后在从左到右校验碰撞, 更新蚂蚁的dstId, 以及其方向
流程图如下 :
这里写图片描述

对于问题二 : 从左开始找出遍历所有的向右的蚂蚁右边的向左的蚂蚁的个数的和即为整个流程的碰撞个数

如下图 : 向右的蚂蚁有3, 11; 3号右边有7, 17, 23三只蚂蚁, 11右边有17, 23两只蚂蚁, 所以整个碰撞的过程中, 碰撞的次数为 3 + 2 = 5 次
这里写图片描述

4. 代码

/*** file name : Test14AntCollition.java* created at : 3:09:41 PM May 30, 2015* created by 970655147*/package com.hx.test04;public class Test14AntCollition {// 蚂蚁爬杆问题// 4.7节讲的是一根长27cm的木棍上,在5个点上有5只蚂蚁,蚂蚁在开始的时候朝任意方向出发,只能掉头或者往前走。// 让任意两只蚂蚁碰头时,它们同时掉头朝反方向走。假设蚂蚁的速度都是一秒一厘米,求蚂蚁都离开木棍的最短时间和最长时间。// 1. 找出所有蚂蚁离开需要的时间// 2. 找出整个流程中碰撞的次数// 3. 找出每一个蚂蚁离开竹竿需要的时间public static void main(String []args ) {int bambooLength = 27;Ant[] ants = new Ant[5];int[] poses = new int[] {3, 7, 11, 17, 23 };boolean[] isRights = new boolean[] {true, false, true, false, false };for(int i=0; i<ants.length; i++) {ants[i] = new Ant(i, poses[i], isRights[i]);}Arrays.sort(ants);Log.log(ants);Bamboo bamboo = new Bamboo(ants, bambooLength);bamboo.findAllClearSpent();bamboo.collisionCount();bamboo.getSpentTimeFor(4);Log.log(bamboo.reallySpendTime );}// 竹竿static class Bamboo {// 蚂蚁(们)[蚂蚁的位置 是经过排序的], 竹竿的长度, 每一只蚂蚁出杆需要的时间[假设杆上只有一只蚂蚁]Ant[] ants;int length;int[] spendTime;int[] reallySpendTime;// 初始化public Bamboo() {}// 排序 确保ants有序public Bamboo(Ant[] ants, int length) {this.ants = ants;this.length = length;spendTime = new int[ants.length];calcSpendTime();}// 找出所有蚂蚁离开的时间// 思路 : 从左边找到第一个向右走的蚂蚁, 从右边找到第一个想左走的蚂蚁// 然后 取这两只蚂蚁花费的时间的较大者public void findAllClearSpent() {int left = 0, right = ants.length - 1;for(left=0; left<ants.length; left++) {if(ants[left].isRight ) {break;}}for(right=ants.length-1; right>=0; right--) {if(!ants[right].isRight ) {break;}}int leftSpent = -1, rightSpent = -1;if(left < ants.length) {leftSpent = length - ants[left].pos / ants[left].speed;}if(right >= 0) {rightSpent = ants[right].pos / ants[right].speed;}Log.log(max(leftSpent, rightSpent) );}// 统计所有蚂蚁的碰撞次数// 思路 : 从右开始遍历ants  每一个向左的ant  会与其左边的向右的所有ant碰一次 // 从而累加得到碰撞次数public void collisionCount() {int cnt = 0;for(int i=ants.length-1; i>=0; i--) {if(!ants[i].isRight ) {for(int j=i-1; j>=0; j--) {if(ants[j].isRight) {cnt ++;}}}}Log.log("collision count : " + cnt);}// 获取指定索引的蚂蚁的出去的时间public void getSpentTimeFor(int idx) {if(reallySpendTime == null) {scanForProcedure();}Log.log(reallySpendTime[idx] );}// 扫描所有的碰撞[一次一次的扫描]// 不断的碰撞 直到所有的蚂蚁都出去了// 先清理头尾可以直接出去的蚂蚁 [左边的向左的, 右边的向右的]// 校验剩余蚂蚁的碰撞   只校验当前情况下课碰撞的蚂蚁  如果发生碰撞更新两只蚂蚁的dstId[交换]  更新方向 // 进入下一个循环迭代// 比如   true false true false false  只校验(0, 1)和 (2, 3)// ants 中存放的是this.ants的副本[确保有序]// output 存放出去的蚂蚁的序列private void scanForProcedure() {Deque<Ant> ants = new LinkedList<Ant>();for(int i=0; i<this.ants.length; i++) {ants.add(new Ant(this.ants[i]) );}List<Ant> output = new ArrayList<Ant>(this.ants.length);// 清理头尾的可以直接出去的蚂蚁bigLoop :while(ants.size() > 0) {while(!ants.getFirst().isRight ) {output.add(ants.removeFirst() );if(ants.size() == 0) {break bigLoop;}}while(ants.getLast().isRight) {output.add(ants.removeLast() );if(ants.size() == 0) {break bigLoop;}}// 校验碰撞Iterator<Ant> it = ants.iterator();Ant last = null;while(it.hasNext() ) {Ant ant = it.next();if(last != null) {if((! ant.isRight) && last.isRight) {collision(last, ant);last = null;continue ;}}last = ant;}}// 设置reallySpendTimereallySpendTime = new int[this.ants.length];for(int i=0; i<this.ants.length; i++) {reallySpendTime[output.get(i).srcId] = spendTime[output.get(i).dstId];}}// 如果发生了碰撞  更新src, dst的dstId[交换], 并更新src, dst的方向private void collision(Ant src, Ant dst) {int tmp = src.dstId;src.dstId = dst.dstId;dst.dstId = tmp;src.isRight = !src.isRight;dst.isRight = !dst.isRight;}// 计算每一只蚂蚁出杆的时间[假设杆上只有一只蚂蚁]private void calcSpendTime() {for(int i=0; i<ants.length; i++) {if(ants[i].isRight ) {spendTime[i] = (length - ants[i].pos) / ants[i].speed;} else {spendTime[i] = ants[i].pos / ants[i].speed;}}}// 获取x, y之间的较大者private static int max(int x, int y) {return x > y ? x : y;}}// 蚂蚁static class Ant implements Comparable {// 蚂蚁的id, dstId表示与其相碰的蚂蚁的id// 位置, 方向是否向右, 速度int srcId;int dstId;int pos;boolean isRight;int speed;// 初始化public Ant() {speed = 1;}public Ant(Ant dst) {this(dst.srcId, dst.pos, dst.isRight);}public Ant(int srcId, int pos, boolean isRight) {this();this.srcId = srcId;this.dstId = srcId;this.pos = pos;this.isRight = isRight;}// compareTo 用于Arrays.sort() 来排序// 这里以pos为基准  谁的pos大  谁就大public int compareTo(Object o) {if(!(o instanceof Ant) ) {return -1;}Ant ant = (Ant) o;return pos - ant.pos;}// Debugpublic String toString() {return "srcId : " + srcId + ", dstId : " + dstId +  ", pos : " + pos + ", isRight : " + isRight;}}}

5. 运行结果

这里写图片描述

6. 总结

这里就体现了想象的重要性, 将一个问题, 转换为另一个问题, 如果没有”将两个蚂蚁的碰撞转换为两个蚂蚁的擦肩而过“的转换, 估计这道题目会很难

注 : 因为作者的水平有限,必然可能出现一些bug, 所以请大家指出!

这篇关于4.7 蚂蚁爬杆问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1011024

相关文章

Flask解决指定端口无法生效问题

《Flask解决指定端口无法生效问题》文章讲述了在使用PyCharm开发Flask应用时,启动地址与手动指定的IP端口不一致的问题,通过修改PyCharm的运行配置,将Flask项目的运行模式从Fla... 目录android问题重现解决方案问题重现手动指定的IP端口是app.run(host='0.0.

Seata之分布式事务问题及解决方案

《Seata之分布式事务问题及解决方案》:本文主要介绍Seata之分布式事务问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Seata–分布式事务解决方案简介同类产品对比环境搭建1.微服务2.SQL3.seata-server4.微服务配置事务模式1

mysql关联查询速度慢的问题及解决

《mysql关联查询速度慢的问题及解决》:本文主要介绍mysql关联查询速度慢的问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql关联查询速度慢1. 记录原因1.1 在一次线上的服务中1.2 最终发现2. 解决方案3. 具体操作总结mysql

一文教你解决Python不支持中文路径的问题

《一文教你解决Python不支持中文路径的问题》Python是一种广泛使用的高级编程语言,然而在处理包含中文字符的文件路径时,Python有时会表现出一些不友好的行为,下面小编就来为大家介绍一下具体的... 目录问题背景解决方案1. 设置正确的文件编码2. 使用pathlib模块3. 转换路径为Unicod

Spring MVC跨域问题及解决

《SpringMVC跨域问题及解决》:本文主要介绍SpringMVC跨域问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录跨域问题不同的域同源策略解决方法1.CORS2.jsONP3.局部解决方案4.全局解决方法总结跨域问题不同的域协议、域名、端口

SpringBoot自定义注解如何解决公共字段填充问题

《SpringBoot自定义注解如何解决公共字段填充问题》本文介绍了在系统开发中,如何使用AOP切面编程实现公共字段自动填充的功能,从而简化代码,通过自定义注解和切面类,可以统一处理创建时间和修改时间... 目录1.1 问题分析1.2 实现思路1.3 代码开发1.3.1 步骤一1.3.2 步骤二1.3.3

基于.NET编写工具类解决JSON乱码问题

《基于.NET编写工具类解决JSON乱码问题》在开发过程中,我们经常会遇到JSON数据处理的问题,尤其是在数据传输和解析过程中,很容易出现编码错误导致的乱码问题,下面我们就来编写一个.NET工具类来解... 目录问题背景核心原理工具类实现使用示例总结在开发过程中,我们经常会遇到jsON数据处理的问题,尤其是

springboot3.4和mybatis plus的版本问题的解决

《springboot3.4和mybatisplus的版本问题的解决》本文主要介绍了springboot3.4和mybatisplus的版本问题的解决,主要由于SpringBoot3.4与MyBat... 报错1:spring-boot-starter/3.4.0/spring-boot-starter-

在 Spring Boot 中使用异步线程时的 HttpServletRequest 复用问题记录

《在SpringBoot中使用异步线程时的HttpServletRequest复用问题记录》文章讨论了在SpringBoot中使用异步线程时,由于HttpServletRequest复用导致... 目录一、问题描述:异步线程操作导致请求复用时 Cookie 解析失败1. 场景背景2. 问题根源二、问题详细分

解读为什么@Autowired在属性上被警告,在setter方法上不被警告问题

《解读为什么@Autowired在属性上被警告,在setter方法上不被警告问题》在Spring开发中,@Autowired注解常用于实现依赖注入,它可以应用于类的属性、构造器或setter方法上,然... 目录1. 为什么 @Autowired 在属性上被警告?1.1 隐式依赖注入1.2 IDE 的警告: