Bootstrap

算法图解——读书笔记02

内存的工作原理

假设你去看演出,需要将东西寄存。寄存处有一个柜子,柜子有很多抽屉。
在这里插入图片描述
每个抽屉可放一样东西,你有两样东西要寄存,因此要了两个抽屉。
在这里插入图片描述
你将两样东西存放在这里。
在这里插入图片描述
现在你可以去看演出了!这大致就是计算机内存的工作原理。计算机就像是很多抽屉的集合体,每个抽屉都有地址。
在这里插入图片描述
fe0ffeeb是一个内存单元的地址。

需要将数据存储到内存时,你请求计算机提供存储空间,计算机给你一个存储地址。需要存储多项数据时,有两种基本方式——数组和链表。

数组
使用数组意味着所有待办事项在内存中都是相连的(紧靠在一起的)。
在这里插入图片描述
现在假设你要添加第四个待办事项,但后面的那个抽屉放着别人的东西!
在这里插入图片描述
在这种情况下,你需要请求计算机重新分配一块可容纳4个待办事项的内存,再将所有待办事项都移到那里。

如果又来了一个待办任务,而当前的地方也没有空位,你们就得再次转移!真是太麻烦了。同样,在数组中添加新元素也可能很麻烦。如果没有了空间,就得移到内存的其他地方,因此添加新元素的速度会很慢。一种解决之道是“预留座位”:即便当前只有3个待办事项,也请计算机提供10个位置,以防需要添加待办事项。这样,只要待办事项不超过10个,就无需转移。这是一个不错的权变措施,但你应该明白,它存在如下两个缺点。

❑ 你额外请求的位置可能根本用不上,这将浪费内存。你没有使用,别人也用不了。
❑ 待办事项超过10个后,你还得转移。

链表
链表中的元素可存储在内存的任何地方。
在这里插入图片描述
链表的每个元素都存储了下一个元素的地址,从而使一系列随机的内存地址串在一起。
在这里插入图片描述
这犹如寻宝游戏。你前往第一个地址,那里有一张纸条写着“下一个元素的地址为123”。因此,你前往地址123,那里又有一张纸条,写着“下一个元素的地址为847”,以此类推。在链表中添加元素很容易:只需将其放入内存,并将其地址存储到前一个元素中

链表的优势在插入元素方面,那数组的优势又是什么呢?

需要随机地读取元素时,数组的效率很高,因为可迅速找到数组的任何元素。在链表中,元素并非靠在一起的,你无法迅速计算出第五个元素的内存地址,而必须先访问第一个元素以获取第二个元素的地址,再访问第二个元素以获取第三个元素的地址,以此类推,直到访问第五个元素。

需要同时读取所有元素时,链表的效率很高:你读取第一个元素,根据其中的地址再读取第二个元素,以此类推。但如果你需要跳跃,链表的效率真的很低。数组与此不同:你知道其中每个元素的地址。例如,假设有一个数组,它包含五个元素,起始地址为00,那么元素#5的地址是多少呢?
在这里插入图片描述
只需执行简单的数学运算就知道:04。

数组的元素带编号,编号从0而不是1开始。例如,在下面的数组中,元素20的位置为1。
在这里插入图片描述
元素的位置称为索引。因此,不说“元素20的位置为1”,而说“元素20位于索引1处”。

下面列出了常见的数组和链表操作的运行时间。
在这里插入图片描述
在中间插入
需要在中间插入元素时,数组和链表哪个更好呢?使用链表时,插入元素很简单,只需修改它前面的那个元素指向的地址。
在这里插入图片描述
而使用数组时,则必须将后面的元素都向后移。
在这里插入图片描述
如果没有足够的空间,可能还得将整个数组复制到其他地方!因此,当需要在中间插入元素时,链表是更好的选择。

删除
如果你要删除元素呢?链表也是更好的选择,因为只需修改前一个元素指向的地址即可。而使用数组时,删除元素后,必须将后面的元素都向前移。

下面是常见数组和链表操作的运行时间。

在这里插入图片描述
数组和链表哪个用的更好呢?显然要看情况,但数组用的很多,因为它支持随机访问。有两种访问方式:随机访问和顺序访问。顺序访问意味着从第一个元素开始逐个地读取元素。链表只能顺序访问:要读取链表的第十个元素,得先读取前九个元素,并沿链接找到第十个元素,随机访问意味着可直接跳到第十个元素。

链表擅长插入和删除,而数组擅长随即机访问

选择排序

假设你的计算机存储了很多乐曲。对于每个乐队,你都记录了其作品被播放的次数。
在这里插入图片描述
你要将这个列表按播放次数从多到少的顺序排列,从而将你喜欢的乐队排序,该如何做呢?

一种办法是遍历这个列表,找出作品播放次数最多的乐队,并将该乐队加到一个新列表中。
在这里插入图片描述
再次这样做,找出播放次数第二多的乐队。
在这里插入图片描述
继续这样做,你将得到一个有序列表。
在这里插入图片描述
下面从计算机科学的角度出发,看看这需要多长时间。别忘了,O(n)时间意味着查看列表中的每个元素一次。例如,对乐队列表进行简单查找时,意味着每个乐队都要查看一次。
在这里插入图片描述
要找出播放次数最多的乐队,必须检查列表中的每个元素。正如你刚才看到的,这需要的时间为O(n)。因此对于这种时间为O(n)的操作,你需要执行n次。
在这里插入图片描述
需要的总时间为O(n × n),即O(n2)。排序算法很有用。

随着排序的进行,每次需要检查的元素数在逐渐减少,最后一次需要检查的元素都只有一个。既然如此,运行时间怎么还是O(n2)呢?这个问题问得好,这与大O表示法中的常数相关。

你说得没错,并非每次都需要检查n个元素。第一次需要检查n个元素,但随后检查的元素数依次为n -1, n -2, …, 2和1。平均每次检查的元素数为1/2 ×n,因此运行时间为O(n×1/2×n)。但大O表示法省略诸如1/2这样的常数,因此简单地写作O(n×n)或O(n2)。

选择排序是一种灵巧的算法,但其速度不是很快。

总结:
❑ 计算机内存犹如一大堆抽屉。
❑ 需要存储多个元素时,可使用数组或链表。
❑ 数组的元素都在一起。
❑ 链表的元素是分开的,其中每个元素都存储了下一个元素的地址。
❑ 数组的读取速度很快。
❑ 链表的插入和删除速度很快。
❑ 在同一个数组中,所有元素的类型都必须相同(都为int、double等)。

;