首页 > 编程语言 >算法学习 | 从几个有趣的故事说起,聊聊里面的算法

算法学习 | 从几个有趣的故事说起,聊聊里面的算法

时间:2023-05-24 10:07:18浏览次数:36  
标签:麦子 格子 故事 兔子 算法 聊聊 对数 有趣

前言

提到故事我就来劲头了。一方面,我喜欢读故事、讲故事、搜集故事,另一方面,用讲故事的方式会为学习增加一些趣味性,有兴趣可以帮助坚持下去。

下面要介绍的故事,有些大家应该不陌生。我之前有读到过,但是没有认真的研究过,有种熟悉的陌生感。

今天分享读了的故事、研究了的解题过程、顺便总结的一些算法知识点和经验。


故事来咯

棋与麦子

故事内容

传说西塔发明了国际象棋而使国王十分高兴,他决定要重赏西塔,西塔说:“我不要您的重赏 ,陛下,只要您在我的棋盘上赏一些麦子就行了。在棋盘的第1个格子里放1粒,在第2个格子里放2粒,在第3个格子里放4粒,在第4个格子里放8粒,依此类推,以后每一个格子里放的麦粒数都是前一个格子里放的麦粒数的2倍,直到放满第64个格子就行了”。国王觉得几粒麦子,这要求很简单,满口答应着,令人如数付给西塔。

摆放麦子的工作开始了,没多久一袋麦子就空了。一袋又一袋的麦子被扛到国王面前来。但是,麦粒数一格接一格飞快增长着,国王很快就看出来,即便拿出全国的粮食,也兑现不了他对西塔的诺言。

所以64个格子到底能放多少粒麦子呢?

解题过程

假设总和是S,那么计算表达式是这样的:

算法学习 | 从几个有趣的故事说起,聊聊里面的算法_斐波那契

将等式①等式左右两侧同时乘以2,等式依旧成立:

算法学习 | 从几个有趣的故事说起,聊聊里面的算法_菲波那切数列_02

好,现在将两个等会进行相减,会得到下面的等式:

算法学习 | 从几个有趣的故事说起,聊聊里面的算法_菲波那切数列_03

我用计算机算了一下S最终的值是18446744073709551615。

这么多粒麦子多重呢?百度中查到的小麦为0.023-0.058克左右,取个平均数吧,0.041克。

18446744073709551615✖️0.041 = 75631650702209152(克)

转换一下单位

≈ 7563 (亿吨)

这个数据相当可观,对比感受一下它大到什么程度。

据国家公布的数据,我们已经连续5年时间小麦年产量达到1.3亿吨。

书中提到这种函数被称为爆炸增量函数。时间复杂度是O(

算法学习 | 从几个有趣的故事说起,聊聊里面的算法_斐波那契_04

)。想想,随着n的值不断增加,这个算法会怎么样?比如购物网站的下单系统,使用人数不断增加,页面可能会出现白屏、无法完成支付等问题。

O(

算法学习 | 从几个有趣的故事说起,聊聊里面的算法_斐波那契_05

)也就是指数时间复杂度的运行效率极差,这样是算法要谨慎使用。

在设计算法的时候,一定要注意算法复杂度增量的问题,尽可能避免使用爆炸增量函数。

兔子数列

故事内容

一般而言,兔子在出生两个月后,就有繁殖能力,一对兔子每个月能生出一对小兔子来。如果所有兔子都不死:

第一个月小兔子没有繁殖能力,所以还是一对;

两个月后,生下一对小兔对数共有两对;

三个月以后,老兔子又生下一对,因为小兔子还没有繁殖能力,所以一共是三对;

......

那么12个月之后会有多少兔子呢?

解题过程

首先,先研究一下兔子出生和成熟规律,这个规律借助表格更加直观:

月数

0

1

2

3

4

5

6

7

8

9

10

11

成兔对数

0

1

1

2

3

5

8

13

21

34

55

89

初生对数

1

0

1

1

2

3

5

8

13

21

34

55

总对数

1

1

2

3

5

8

13

21

34

55

89

144

从第二个月开始,就有规律了,每过一个月,初生兔子变成熟,成熟兔子生出一对兔子,就有这样的等式:

  • 初生对数=前一月的成兔对数
  • 成兔对数=前一月的成兔对数+前一月的初生对数
  • 总体对数=本月的成兔对数+本月的初生对数

这么有趣的规律,数学家就开始整活了,数学家莱昂纳多·斐波那契以兔子繁殖为例子引入了菲波那切数列,又称为“兔子数列”。

菲波那切数列如下:

1、1、2、3、5、8、13、21、34、55、89……

递归表达式:

算法学习 | 从几个有趣的故事说起,聊聊里面的算法_时间复杂度_06

重点来了,这个算法如何设计?

function fibFunc(n) {
  if (n < 1) {
    return 1;
  }
  if (n === 1 || n === 2) {
    return 1;
  }
  let f1 = 1;
  let f2 = 1;
  for (let i = 3; i <= n; i++) {
    f2 = f2 + f1; // 辗转相加法
    f1 = f2 - f1; // 记录前一项
  }
  return f2;
}
console.log(fibFunc(1)); // 1
console.log(fibFunc(2)); // 1
console.log(fibFunc(3)); // 2
console.log(fibFunc(11)); // 89

该算法的复杂性为:

空间复杂度:O(n)

时间复杂度:O(1)

未完待续

果然故事有趣,学起来也轻松不少。

还有一个更有趣的故事,叫做自然界里关于斐波那契的巧合。精炼一下这个故事就是

树木抽长新枝条的规律、一些花的花瓣数量,都符合斐波那契数列。而这种自然现象,是因为这些植物按照自然的规律才进化成这样。似乎是植物排列种子的“优化方式”,便于更好的生存和生长。

我有种不但逐渐熟悉了算法,还学习了自然相关的新知识的双重收获的快乐感觉。

老实说,除了故事本身,逐渐找到规律、研究规律,并找到对应的算法,真是一件非常愉快的事。

对算法的学习,逐渐上头。

标签:麦子,格子,故事,兔子,算法,聊聊,对数,有趣
From: https://blog.51cto.com/u_15838863/6336860

相关文章

  • Algorithm_01--C#递归算法
    ///递归算法本质:///1、方法的自我调用///2、有明确的终止条件///3、每次调用时,问题规模在不断减少。通过递减,最终到达终止条件  问题:程序在输入1000后(即1到1000的和),程序会出现异常。解答:百度后得出结论,栈溢出异常。1、递归......
  • TPSO-DSDT粒子群算法在三维装箱问题上的应用
    组合算法是将传统启发式算法与数学规划算法结合元启发式算法共同工作进行相应的计算,还有融合多种算法所获得的计算方法,结合了所有算法自身的有点,规避其自身缺点从而达到解决装箱问题的最终目的。现在,组合算法的整体规划绝大多数都是通过启发式算法完成的,局部优化的过程采用的是人......
  • 十大经典排序算法总结
    排序算法可以分为:内部排序:数据记录在内存中进行排序。外部排序:因排序的数据很大,内存不能容纳全部的排序记录,在排序过程中需要访问外存。常见的内部排序算法有:插入排序、希尔排序、选择排序、冒泡排序、归并排序、快速排序、堆排序、基数排序、计数排序、桶排序。其中比较类......
  • Day_01--C#递归算法
     ///递归算法本质:///1、方法的自我调用///2、有明确的终止条件///3、每次调用时,问题规模在不断减少。通过递减,最终到达终止条件 问题:程序在输入1000后(即1到1000的和),程序会出现异常。解答:百度后得出结论,栈溢出异常。1、递归方法在每......
  • 代码随想录算法训练营第十四天|144. 二叉树的前序遍历、145. 二叉树的后序遍历、94.
    【参考链接】1.满二叉树,完全二叉树,二叉搜索树(有数值,有序树)。2.平衡二叉搜索树:又被称为AVL(Adelson-VelskyandLandis)树,且具有以下性质:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。3.优先级队列其实是一个堆,堆就是一棵完全二叉......
  • 聊聊ElasticeSearch并发写的乐观锁机制
    概述ES的多客户端并发更新是基于乐观并发控制,通过版本号机制来实现冲突检测。关键对象ES的老版本是用过_version字段的版本号实现乐观锁的。现在新版增加了基于_seq_no与_primary_term字段,三个字段做乐观锁并发控制。_version:标识文档的版本号,只有当前文档的更新,该字段才会累......
  • 算法刷题记录:NC22227 约瑟夫环
    题目链接https://ac.nowcoder.com/acm/problem/22227解题思路模拟环。这道题顺序数就行,顺序是逆时针,逆时针的箭头是往左拐的,变成直线后趋于正半轴所以是+。不过,这道模拟环并没有说从idx号开始,往左/右数几个人,所以不需要考虑+或-。因为不会越界,所以也不用额外%n。AC代码......
  • 基于PSO优化的SVM数据预测算法matlab仿真
    1.算法仿真效果matlab2022a仿真结果如下:     2.算法涉及理论知识概要         支持向量机(supportvectormachines,SVM)是二分类算法,所谓二分类即把具有多个特性(属性)的数据分为两类,目前主流机器学习算法中,神经网络等其他机器学习模型已经能很好完成二分......
  • m基于matlab的LDPC译码算法性能仿真,对比BP译码,最小和译码以及归一化偏移最小和译码
    1.算法仿真效果matlab2022a仿真结果如下:  2.算法涉及理论知识概要       LDPC码是麻省理工学院RobertGallager于1963年在博士论文中提出的一种具有稀疏校验矩阵的分组纠错码。几乎适用于所有的信道,因此成为编码界近年来的研究热点。它的性能逼近香农极限,且描述和......
  • 启发式算法在三维装箱问题上的应用
    启发式算法的出现时间比较难以确定,因为很多算法的提出都是在不同的领域和不同的时间段内,而且随着时间的推移,这些算法也在不断地完善和发展。以下是一些比较有代表性的启发式算法及其出现时间:1953年,模拟退火算法(SimulatedAnnealing,SA)模拟退火算法是一种基于固体物理学中固体退火......