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

相关文章

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

解决Cron定时任务中Pytest脚本无法发送邮件的问题

《解决Cron定时任务中Pytest脚本无法发送邮件的问题》文章探讨解决在Cron定时任务中运行Pytest脚本时邮件发送失败的问题,先优化环境变量,再检查Pytest邮件配置,接着配置文件确保SMT... 目录引言1. 环境变量优化:确保Cron任务可以正确执行解决方案:1.1. 创建一个脚本1.2. 修

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g

SpringBoot项目删除Bean或者不加载Bean的问题解决

《SpringBoot项目删除Bean或者不加载Bean的问题解决》文章介绍了在SpringBoot项目中如何使用@ComponentScan注解和自定义过滤器实现不加载某些Bean的方法,本文通过实... 使用@ComponentScan注解中的@ComponentScan.Filter标记不加载。@C