This commit is contained in:
krahets
2023-04-07 22:32:13 +08:00
parent c515efea8b
commit 54b72a3ac9
14 changed files with 226 additions and 229 deletions

View File

@ -1768,15 +1768,11 @@
<h1 id="41">4.1. &nbsp; 数组<a class="headerlink" href="#41" title="Permanent link">&para;</a></h1>
<p>「数组 Array」是一种<strong>相同类型元素</strong> 存储在 <strong>连续内存空间</strong> 的数据结构,将元素在数组中的位置称为元素的「索引 Index」。</p>
<p>「数组 Array」是一种线性数据结构,其将相同类型元素存储在连续内存空间中。我们将元素在数组中的位置称为元素的「索引 Index」。</p>
<p><img alt="数组定义与存储方式" src="../array.assets/array_definition.png" /></p>
<p align="center"> Fig. 数组定义与存储方式 </p>
<div class="admonition note">
<p class="admonition-title">Note</p>
<p>观察上图,我们发现 <strong>数组首元素的索引为 <span class="arithmatex">\(0\)</span></strong> 。你可能会想,这并不符合日常习惯,首个元素的索引为什么不是 <span class="arithmatex">\(1\)</span> 呢,这不是更加自然吗?我认同你的想法,但请先记住这个设定,后面讲内存地址计算时,我会尝试解答这个问题。</p>
</div>
<p><strong>数组初始化</strong>。一般会用到无初始值、给定初始值两种写法,可根据需求选取。在不给定初始值的情况下,一般所有元素会被初始化为默认值 <span class="arithmatex">\(0\)</span></p>
<p><strong>数组初始化</strong>。通常有无初始值和给定初始值两种方式,我们可根据需求选择合适的方法。在未给定初始值的情况下,数组的所有元素通常会被初始化为默认值 <span class="arithmatex">\(0\)</span></p>
<div class="tabbed-set tabbed-alternate" data-tabs="1:10"><input checked="checked" id="__tabbed_1_1" name="__tabbed_1" type="radio" /><input id="__tabbed_1_2" name="__tabbed_1" type="radio" /><input id="__tabbed_1_3" name="__tabbed_1" type="radio" /><input id="__tabbed_1_4" name="__tabbed_1" type="radio" /><input id="__tabbed_1_5" name="__tabbed_1" type="radio" /><input id="__tabbed_1_6" name="__tabbed_1" type="radio" /><input id="__tabbed_1_7" name="__tabbed_1" type="radio" /><input id="__tabbed_1_8" name="__tabbed_1" type="radio" /><input id="__tabbed_1_9" name="__tabbed_1" type="radio" /><input id="__tabbed_1_10" name="__tabbed_1" type="radio" /><div class="tabbed-labels"><label for="__tabbed_1_1">Java</label><label for="__tabbed_1_2">C++</label><label for="__tabbed_1_3">Python</label><label for="__tabbed_1_4">Go</label><label for="__tabbed_1_5">JavaScript</label><label for="__tabbed_1_6">TypeScript</label><label for="__tabbed_1_7">C</label><label for="__tabbed_1_8">C#</label><label for="__tabbed_1_9">Swift</label><label for="__tabbed_1_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -1848,15 +1844,19 @@
</div>
</div>
<h2 id="411">4.1.1. &nbsp; 数组优点<a class="headerlink" href="#411" title="Permanent link">&para;</a></h2>
<p><strong>在数组中访问元素非常高效</strong>这是因为在数组中,计算元素的内存地址非常容易。给定数组首个元素的地址、和一个元素的索引,用以下公式可以直接计算得到该元素的内存地址,从而直接访问此元素。</p>
<p><strong>在数组中访问元素非常高效</strong>由于数组元素被存储在连续的内存空间中,因此计算数组元素的内存地址非常容易。给定数组首个元素的地址和某个元素的索引,我们可以使用以下公式计算得到该元素的内存地址,从而直接访问此元素。</p>
<p><img alt="数组元素的内存地址计算" src="../array.assets/array_memory_location_calculation.png" /></p>
<p align="center"> Fig. 数组元素的内存地址计算 </p>
<div class="highlight"><pre><span></span><code><a id="__codelineno-10-1" name="__codelineno-10-1" href="#__codelineno-10-1"></a><span class="c1"># 元素内存地址 = 数组内存地址 + 元素长度 * 元素索引</span>
<a id="__codelineno-10-2" name="__codelineno-10-2" href="#__codelineno-10-2"></a><span class="nv">elementAddr</span><span class="w"> </span><span class="o">=</span><span class="w"> </span>firtstElementAddr<span class="w"> </span>+<span class="w"> </span>elementLength<span class="w"> </span>*<span class="w"> </span>elementIndex
</code></pre></div>
<p><strong>为什么数组元素索引从 0 开始编号?</strong> 根据地址计算公式,<strong>索引本质上表示的是内存地址偏移量</strong>,首个元素的地址偏移量是 <span class="arithmatex">\(0\)</span> ,那么索引是 <span class="arithmatex">\(0\)</span> 也就很自然了。</p>
<p>访问元素的高效性带来了许多便利。例如,我们可以在 <span class="arithmatex">\(O(1)\)</span> 时间内随机获取一个数组中的元素。</p>
<div class="admonition question">
<p class="admonition-title">为什么数组元素的索引要从 <span class="arithmatex">\(0\)</span> 开始编号呢?</p>
<p>观察上图,我们发现数组首个元素的索引为 <span class="arithmatex">\(0\)</span> ,这似乎有些反直觉,因为从 <span class="arithmatex">\(1\)</span> 开始计数会更自然。</p>
<p>然而,从地址计算公式的角度看,<strong>索引本质上表示的是内存地址的偏移量</strong>。首个元素的地址偏移量是 <span class="arithmatex">\(0\)</span> ,因此索引为 <span class="arithmatex">\(0\)</span> 也是合理的。</p>
</div>
<p>访问元素的高效性带来了诸多便利。例如,我们可以在 <span class="arithmatex">\(O(1)\)</span> 时间内随机获取数组中的任意一个元素。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="2:10"><input checked="checked" id="__tabbed_2_1" name="__tabbed_2" type="radio" /><input id="__tabbed_2_2" name="__tabbed_2" type="radio" /><input id="__tabbed_2_3" name="__tabbed_2" type="radio" /><input id="__tabbed_2_4" name="__tabbed_2" type="radio" /><input id="__tabbed_2_5" name="__tabbed_2" type="radio" /><input id="__tabbed_2_6" name="__tabbed_2" type="radio" /><input id="__tabbed_2_7" name="__tabbed_2" type="radio" /><input id="__tabbed_2_8" name="__tabbed_2" type="radio" /><input id="__tabbed_2_9" name="__tabbed_2" type="radio" /><input id="__tabbed_2_10" name="__tabbed_2" type="radio" /><div class="tabbed-labels"><label for="__tabbed_2_1">Java</label><label for="__tabbed_2_2">C++</label><label for="__tabbed_2_3">Python</label><label for="__tabbed_2_4">Go</label><label for="__tabbed_2_5">JavaScript</label><label for="__tabbed_2_6">TypeScript</label><label for="__tabbed_2_7">C</label><label for="__tabbed_2_8">C#</label><label for="__tabbed_2_9">Swift</label><label for="__tabbed_2_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2321,9 +2321,9 @@
</div>
<p>总结来看,数组的插入与删除操作有以下缺点:</p>
<ul>
<li><strong>时间复杂度高</strong>:数组的插入和删除的平均时间复杂度均为 <span class="arithmatex">\(O(N)\)</span> ,其中 <span class="arithmatex">\(N\)</span> 为数组长度。</li>
<li><strong>丢失元素</strong>:由于数组的长度不可变,因此在插入元素后,超出数组长度范围的元素会丢失。</li>
<li><strong>内存浪费</strong>:我们一般会初始化一个比较长的数组,只用前面一部分,这样在插入数据时,丢失的末尾元素都是我们不关心的,但这样做同时也会造成内存空间的浪费。</li>
<li><strong>时间复杂度高</strong>:数组的插入和删除的平均时间复杂度均为 <span class="arithmatex">\(O(n)\)</span> ,其中 <span class="arithmatex">\(n\)</span> 为数组长度。</li>
<li><strong>丢失元素</strong>:由于数组的长度不可变,因此在插入元素后,超出数组长度范围的元素会丢失。</li>
<li><strong>内存浪费</strong>:我们可以初始化一个比较长的数组,只用前面一部分,这样在插入数据时,丢失的末尾元素都是我们不关心的,但这样做同时也会造成内存空间的浪费。</li>
</ul>
<h2 id="413">4.1.3. &nbsp; 数组常用操作<a class="headerlink" href="#413" title="Permanent link">&para;</a></h2>
<p><strong>数组遍历</strong>。以下介绍两种常用的遍历方法。</p>

View File

@ -1768,9 +1768,9 @@
<h1 id="42">4.2. &nbsp; 链表<a class="headerlink" href="#42" title="Permanent link">&para;</a></h1>
<p>内存空间是所有程序的公共资源,排除已被占用的内存空间,空闲内存空间往往是散落在内存各处的。上节讲到,<strong>存储数组的内存空间必须是连续的</strong>当我们需要申请一个非常大的数组时,系统不一定能够分配这么大的连续内存空间。</p>
<p>相对地,链表则更加灵活,可以存储非连续的内存空间。「链表 Linked List」是一种线性数据结构每个元素都是单独的对象,各个元素(即结点之间通过指针连接。由于结点中记录了连接关系,因此链表的存储方式相比于数组更加灵活,系统可将结点分散在内存各处,而不必保证内存地址的连续性</p>
<p>链表「结点 Node」包含两项数据一是结点「值 Value」二是指向下一结点的「指针 Pointer」或称「引用 Reference」</p>
<p>内存空间是所有程序的公共资源,排除已被占用的内存空间,空闲内存空间通常散落在内存各处。在上一节中,我们提到存储数组的内存空间必须是连续的,而当我们需要申请一个非常大的数组时,空闲内存中可能没有这么大的连续空间。</p>
<p>与数组相比,链表更具灵活性,因为它可以存储非连续的内存空间。「链表 Linked List」是一种线性数据结构其每个元素都是一个结点对象,各个结点之间通过指针连接,从当前结点通过指针可以访问到下一个结点。由于指针记录了下个结点的内存地址,因此无需保证内存地址的连续性,从而可以将各个结点分散存储在内存各处。</p>
<p>链表「结点 Node」包含两项数据一是结点「值 Value」二是指向下一结点的「指针 Pointer」或称指向下一结点的「引用 Reference」。</p>
<p><img alt="链表定义与存储方式" src="../linked_list.assets/linkedlist_definition.png" /></p>
<p align="center"> Fig. 链表定义与存储方式 </p>
@ -1904,12 +1904,15 @@
</div>
</div>
</div>
<p><strong>尾结点指向什么?</strong> 我们一般将链表的最后一个结点称为「尾结点」,其指向的是「空」,在 Java / C++ / Python 中分别记为 <code>null</code> / <code>nullptr</code> / <code>None</code> 。在不引起歧义下,本书都使用 <code>null</code> 来表示空。</p>
<p><strong>链表初始化方法</strong>。建立链表分为两步,第一步是初始化各个结点对象,第二步是构建引用指向关系。完成后,即可以从链表的首个结点(即头结点)出发,访问其余所有的结点。</p>
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p>我们通常将头结点当作链表的代称,例如头结点 <code>head</code> 和链表 <code>head</code> 实际上是同义的。</p>
<div class="admonition question">
<p class="admonition-title">尾结点指向什么?</p>
<p>我们将链表的最后一个结点称为「尾结点」,其指向的是“空”,在 Java, C++, Python 中分别记为 <code>null</code>, <code>nullptr</code>, <code>None</code> 。在不引起歧义的前提下,本书都使用 <code>null</code> 来表示空。</p>
</div>
<div class="admonition question">
<p class="admonition-title">如何称呼链表?</p>
<p>在编程语言中,数组整体就是一个变量,例如数组 <code>nums</code> ,包含各个元素 <code>nums[0]</code> , <code>nums[1]</code> 等等。而链表是由多个结点对象组成,我们通常将头结点当作链表的代称,例如头结点 <code>head</code> 和链表 <code>head</code> 实际上是同义的。</p>
</div>
<p><strong>链表初始化方法</strong>。建立链表分为两步,第一步是初始化各个结点对象,第二步是构建引用指向关系。完成后,即可以从链表的头结点(即首个结点)出发,通过指针 <code>next</code> 依次访问所有结点。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="2:10"><input checked="checked" id="__tabbed_2_1" name="__tabbed_2" type="radio" /><input id="__tabbed_2_2" name="__tabbed_2" type="radio" /><input id="__tabbed_2_3" name="__tabbed_2" type="radio" /><input id="__tabbed_2_4" name="__tabbed_2" type="radio" /><input id="__tabbed_2_5" name="__tabbed_2" type="radio" /><input id="__tabbed_2_6" name="__tabbed_2" type="radio" /><input id="__tabbed_2_7" name="__tabbed_2" type="radio" /><input id="__tabbed_2_8" name="__tabbed_2" type="radio" /><input id="__tabbed_2_9" name="__tabbed_2" type="radio" /><input id="__tabbed_2_10" name="__tabbed_2" type="radio" /><div class="tabbed-labels"><label for="__tabbed_2_1">Java</label><label for="__tabbed_2_2">C++</label><label for="__tabbed_2_3">Python</label><label for="__tabbed_2_4">Go</label><label for="__tabbed_2_5">JavaScript</label><label for="__tabbed_2_6">TypeScript</label><label for="__tabbed_2_7">C</label><label for="__tabbed_2_8">C#</label><label for="__tabbed_2_9">Swift</label><label for="__tabbed_2_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -1965,12 +1968,11 @@
<a id="__codelineno-13-5" name="__codelineno-13-5" href="#__codelineno-13-5"></a><span class="nx">n2</span><span class="w"> </span><span class="o">:=</span><span class="w"> </span><span class="nx">NewListNode</span><span class="p">(</span><span class="mi">2</span><span class="p">)</span>
<a id="__codelineno-13-6" name="__codelineno-13-6" href="#__codelineno-13-6"></a><span class="nx">n3</span><span class="w"> </span><span class="o">:=</span><span class="w"> </span><span class="nx">NewListNode</span><span class="p">(</span><span class="mi">5</span><span class="p">)</span>
<a id="__codelineno-13-7" name="__codelineno-13-7" href="#__codelineno-13-7"></a><span class="nx">n4</span><span class="w"> </span><span class="o">:=</span><span class="w"> </span><span class="nx">NewListNode</span><span class="p">(</span><span class="mi">4</span><span class="p">)</span>
<a id="__codelineno-13-8" name="__codelineno-13-8" href="#__codelineno-13-8"></a>
<a id="__codelineno-13-9" name="__codelineno-13-9" href="#__codelineno-13-9"></a><span class="c1">// 构建引用指向</span>
<a id="__codelineno-13-10" name="__codelineno-13-10" href="#__codelineno-13-10"></a><span class="nx">n0</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n1</span>
<a id="__codelineno-13-11" name="__codelineno-13-11" href="#__codelineno-13-11"></a><span class="nx">n1</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n2</span>
<a id="__codelineno-13-12" name="__codelineno-13-12" href="#__codelineno-13-12"></a><span class="nx">n2</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n3</span>
<a id="__codelineno-13-13" name="__codelineno-13-13" href="#__codelineno-13-13"></a><span class="nx">n3</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n4</span>
<a id="__codelineno-13-8" name="__codelineno-13-8" href="#__codelineno-13-8"></a><span class="c1">// 构建引用指向</span>
<a id="__codelineno-13-9" name="__codelineno-13-9" href="#__codelineno-13-9"></a><span class="nx">n0</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n1</span>
<a id="__codelineno-13-10" name="__codelineno-13-10" href="#__codelineno-13-10"></a><span class="nx">n1</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n2</span>
<a id="__codelineno-13-11" name="__codelineno-13-11" href="#__codelineno-13-11"></a><span class="nx">n2</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n3</span>
<a id="__codelineno-13-12" name="__codelineno-13-12" href="#__codelineno-13-12"></a><span class="nx">n3</span><span class="p">.</span><span class="nx">Next</span><span class="w"> </span><span class="p">=</span><span class="w"> </span><span class="nx">n4</span>
</code></pre></div>
</div>
<div class="tabbed-block">
@ -2066,7 +2068,7 @@
</div>
</div>
<h2 id="421">4.2.1. &nbsp; 链表优点<a class="headerlink" href="#421" title="Permanent link">&para;</a></h2>
<p><strong>链表中插入与删除结点的操作效率高</strong>如,如果我们想在链表中间的两个结点 <code>A</code> , <code>B</code> 之间插入一个新结点 <code>P</code> ,我们只需要改变两个结点指针即可,时间复杂度为 <span class="arithmatex">\(O(1)\)</span> 相比数组的插入操作高效很多。</p>
<p><strong>链表中插入与删除结点的操作效率高</strong>如,如果我们想在链表中间的两个结点 <code>A</code> , <code>B</code> 之间插入一个新结点 <code>P</code> ,我们只需要改变两个结点指针即可,时间复杂度为 <span class="arithmatex">\(O(1)\)</span> 相比之下,数组的插入操作效率要低得多。</p>
<p><img alt="链表插入结点" src="../linked_list.assets/linkedlist_insert_node.png" /></p>
<p align="center"> Fig. 链表插入结点 </p>
@ -2159,7 +2161,7 @@
</div>
</div>
</div>
<p>在链表中删除结点也方便,只需改变一个结点指针即可。如下图所示,虽然在完成删除后结点 <code>P</code> 仍然指向 <code>n1</code> ,但实际上 <code>P</code> 已经不属于此链表,因为遍历此链表无法访问到 <code>P</code></p>
<p>在链表中删除结点也非常方便,只需改变一个结点指针即可。如下图所示,尽管在删除操作完成后,结点 <code>P</code> 仍然指向 <code>n1</code>,但实际上 <code>P</code> 已经不属于此链表,因为遍历此链表无法访问到 <code>P</code></p>
<p><img alt="链表删除结点" src="../linked_list.assets/linkedlist_remove_node.png" /></p>
<p align="center"> Fig. 链表删除结点 </p>
@ -2285,7 +2287,7 @@
</div>
</div>
<h2 id="422">4.2.2. &nbsp; 链表缺点<a class="headerlink" href="#422" title="Permanent link">&para;</a></h2>
<p><strong>链表访问结点效率低</strong>。上节提到,数组可以在 <span class="arithmatex">\(O(1)\)</span> 时间下访问任意元素,链表无法直接访问任意结点这是因为计算机需要从头结点出发,一个一个地向后遍历到目标结点。例如,倘若想要访问链表索引为 <code>index</code> (即第 <code>index + 1</code> 个)的结点,那么需要 <code>index</code> 次访问操作</p>
<p><strong>链表访问结点效率</strong>上节所述,数组可以在 <span class="arithmatex">\(O(1)\)</span> 时间下访问任意元素。然而,链表无法直接访问任意结点这是因为系统需要从头结点出发,逐个向后遍历直至找到目标结点。例如,要访问链表索引为 <code>index</code>(即第 <code>index + 1</code> 个)的结点,则需要向后遍历 <code>index</code> </p>
<div class="tabbed-set tabbed-alternate" data-tabs="5:10"><input checked="checked" id="__tabbed_5_1" name="__tabbed_5" type="radio" /><input id="__tabbed_5_2" name="__tabbed_5" type="radio" /><input id="__tabbed_5_3" name="__tabbed_5" type="radio" /><input id="__tabbed_5_4" name="__tabbed_5" type="radio" /><input id="__tabbed_5_5" name="__tabbed_5" type="radio" /><input id="__tabbed_5_6" name="__tabbed_5" type="radio" /><input id="__tabbed_5_7" name="__tabbed_5" type="radio" /><input id="__tabbed_5_8" name="__tabbed_5" type="radio" /><input id="__tabbed_5_9" name="__tabbed_5" type="radio" /><input id="__tabbed_5_10" name="__tabbed_5" type="radio" /><div class="tabbed-labels"><label for="__tabbed_5_1">Java</label><label for="__tabbed_5_2">C++</label><label for="__tabbed_5_3">Python</label><label for="__tabbed_5_4">Go</label><label for="__tabbed_5_5">JavaScript</label><label for="__tabbed_5_6">TypeScript</label><label for="__tabbed_5_7">C</label><label for="__tabbed_5_8">C#</label><label for="__tabbed_5_9">Swift</label><label for="__tabbed_5_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2408,7 +2410,7 @@
</div>
</div>
</div>
<p><strong>链表的内存占用</strong>。链表以结点为单位,每个结点除了保存值外,还需额外保存指针(引用)。这意味着同样数据量下,链表比数组需要占用更多内存空间。</p>
<p><strong>链表的内存占用较大</strong>。链表以结点为单位,每个结点除了保存值外,还需额外保存指针(引用)。这意味着在相同数据量的情况下,链表比数组需要占用更多内存空间。</p>
<h2 id="423">4.2.3. &nbsp; 链表常用操作<a class="headerlink" href="#423" title="Permanent link">&para;</a></h2>
<p><strong>遍历链表查找</strong>。遍历链表,查找链表内值为 <code>target</code> 的结点,输出结点在链表中的索引。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="6:10"><input checked="checked" id="__tabbed_6_1" name="__tabbed_6" type="radio" /><input id="__tabbed_6_2" name="__tabbed_6" type="radio" /><input id="__tabbed_6_3" name="__tabbed_6" type="radio" /><input id="__tabbed_6_4" name="__tabbed_6" type="radio" /><input id="__tabbed_6_5" name="__tabbed_6" type="radio" /><input id="__tabbed_6_6" name="__tabbed_6" type="radio" /><input id="__tabbed_6_7" name="__tabbed_6" type="radio" /><input id="__tabbed_6_8" name="__tabbed_6" type="radio" /><input id="__tabbed_6_9" name="__tabbed_6" type="radio" /><input id="__tabbed_6_10" name="__tabbed_6" type="radio" /><div class="tabbed-labels"><label for="__tabbed_6_1">Java</label><label for="__tabbed_6_2">C++</label><label for="__tabbed_6_3">Python</label><label for="__tabbed_6_4">Go</label><label for="__tabbed_6_5">JavaScript</label><label for="__tabbed_6_6">TypeScript</label><label for="__tabbed_6_7">C</label><label for="__tabbed_6_8">C#</label><label for="__tabbed_6_9">Swift</label><label for="__tabbed_6_10">Zig</label></div>
@ -2551,9 +2553,9 @@
</div>
</div>
<h2 id="424">4.2.4. &nbsp; 常见链表类型<a class="headerlink" href="#424" title="Permanent link">&para;</a></h2>
<p><strong>单向链表</strong>。即上述介绍的普通链表。单向链表的结点有「值」和指向下一结点的指针(引用)两项数据。我们将首个结点称为头结点,尾结点指向 <code>null</code></p>
<p><strong>环形链表</strong>。如果我们令单向链表的尾结点指向头结点(即首尾相接),则得到一个环形链表。在环形链表中,我们可以将任意结点看作是头结点。</p>
<p><strong>双向链表</strong>。单向链表记录了个方向的指针(引用),在双向链表的结点定义中,同时有指向下一结点(后继结点)和上一结点(前驱结点)的指针(引用)」。双向链表相对于单向链表更灵活,可以朝两个方向遍历链表,但也需要占用更多的内存空间。</p>
<p><strong>单向链表</strong>。即上述介绍的普通链表。单向链表的结点包含值和指向下一结点的指针(引用)两项数据。我们将首个结点称为头结点,将最后一个结点成为尾结点,尾结点指向 <code>null</code></p>
<p><strong>环形链表</strong>。如果我们令单向链表的尾结点指向头结点(即首尾相接),则得到一个环形链表。在环形链表中,任意结点都可以视作头结点。</p>
<p><strong>双向链表</strong>单向链表相比,双向链表记录了个方向的指针(引用)双向链表的结点定义同时包含指向后继结点(下一结点)和前驱结点(上一结点)的指针。相较于单向链表,双向链表更灵活,可以朝两个方向遍历链表,但相应地也需要占用更多的内存空间。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="7:10"><input checked="checked" id="__tabbed_7_1" name="__tabbed_7" type="radio" /><input id="__tabbed_7_2" name="__tabbed_7" type="radio" /><input id="__tabbed_7_3" name="__tabbed_7" type="radio" /><input id="__tabbed_7_4" name="__tabbed_7" type="radio" /><input id="__tabbed_7_5" name="__tabbed_7" type="radio" /><input id="__tabbed_7_6" name="__tabbed_7" type="radio" /><input id="__tabbed_7_7" name="__tabbed_7" type="radio" /><input id="__tabbed_7_8" name="__tabbed_7" type="radio" /><input id="__tabbed_7_9" name="__tabbed_7" type="radio" /><input id="__tabbed_7_10" name="__tabbed_7" type="radio" /><div class="tabbed-labels"><label for="__tabbed_7_1">Java</label><label for="__tabbed_7_2">C++</label><label for="__tabbed_7_3">Python</label><label for="__tabbed_7_4">Go</label><label for="__tabbed_7_5">JavaScript</label><label for="__tabbed_7_6">TypeScript</label><label for="__tabbed_7_7">C</label><label for="__tabbed_7_8">C#</label><label for="__tabbed_7_9">Swift</label><label for="__tabbed_7_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">

View File

@ -739,7 +739,7 @@
<li class="md-nav__item">
<a href="#432" class="md-nav__link">
4.3.2. &nbsp; 列表简易实现 *
4.3.2. &nbsp; 列表实现 *
</a>
</li>
@ -1711,7 +1711,7 @@
<li class="md-nav__item">
<a href="#432" class="md-nav__link">
4.3.2. &nbsp; 列表简易实现 *
4.3.2. &nbsp; 列表实现 *
</a>
</li>
@ -1740,10 +1740,10 @@
<h1 id="43">4.3. &nbsp; 列表<a class="headerlink" href="#43" title="Permanent link">&para;</a></h1>
<p><strong>由于长度不可变,数组的实用性大大降低</strong>。在多情况下,我们事先并不知道会输入多少数据,这就为数组长度的选择带来了很大困难。长度选小了,需要在添加数据频繁扩容数组;长度选大了,又造成内存空间的浪费。</p>
<p>解决此问题,诞生了一种被称为「列表 List」的数据结构。列表可以被理解为长度可变的数组,因此也常被称为「动态数组 Dynamic Array」。列表基于数组实现,继承了数组的优点,同时还可以在程序运行中实时扩容。在列表中,我们可以自由添加元素,而不用担心超过容量限制。</p>
<p><strong>数组长度不可变导致实用性降低</strong>。在多情况下,我们事先无法确定需要存储多少数据,这使数组长度的选择变得困难。长度过小,需要在持续添加数据频繁扩容数组;长度过大,则会造成内存空间的浪费。</p>
<p>为解决此问题,出现了一种被称为「动态数组 Dynamic Array」的数据结构长度可变的数组,也常被称为「列表 List」。列表基于数组实现,继承了数组的优点,并且可以在程序运行过程中动态扩容。在列表中,我们可以自由添加元素,而无需担心超过容量限制。</p>
<h2 id="431">4.3.1. &nbsp; 列表常用操作<a class="headerlink" href="#431" title="Permanent link">&para;</a></h2>
<p><strong>初始化列表</strong>我们通常会使用“无初始值”和“有初始值”的两种初始化方法。</p>
<p><strong>初始化列表</strong>。通常我们会使用“无初始值”和“有初始值”的两种初始化方法。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="1:10"><input checked="checked" id="__tabbed_1_1" name="__tabbed_1" type="radio" /><input id="__tabbed_1_2" name="__tabbed_1" type="radio" /><input id="__tabbed_1_3" name="__tabbed_1" type="radio" /><input id="__tabbed_1_4" name="__tabbed_1" type="radio" /><input id="__tabbed_1_5" name="__tabbed_1" type="radio" /><input id="__tabbed_1_6" name="__tabbed_1" type="radio" /><input id="__tabbed_1_7" name="__tabbed_1" type="radio" /><input id="__tabbed_1_8" name="__tabbed_1" type="radio" /><input id="__tabbed_1_9" name="__tabbed_1" type="radio" /><input id="__tabbed_1_10" name="__tabbed_1" type="radio" /><div class="tabbed-labels"><label for="__tabbed_1_1">Java</label><label for="__tabbed_1_2">C++</label><label for="__tabbed_1_3">Python</label><label for="__tabbed_1_4">Go</label><label for="__tabbed_1_5">JavaScript</label><label for="__tabbed_1_6">TypeScript</label><label for="__tabbed_1_7">C</label><label for="__tabbed_1_8">C#</label><label for="__tabbed_1_9">Swift</label><label for="__tabbed_1_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -1826,7 +1826,7 @@
</div>
</div>
</div>
<p><strong>访问与更新元素</strong>。列表的底层数据结构是数组,因此可以在 <span class="arithmatex">\(O(1)\)</span> 时间内访问更新元素,效率很高。</p>
<p><strong>访问与更新元素</strong>由于列表的底层数据结构是数组,因此可以在 <span class="arithmatex">\(O(1)\)</span> 时间内访问更新元素,效率很高。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="2:10"><input checked="checked" id="__tabbed_2_1" name="__tabbed_2" type="radio" /><input id="__tabbed_2_2" name="__tabbed_2" type="radio" /><input id="__tabbed_2_3" name="__tabbed_2" type="radio" /><input id="__tabbed_2_4" name="__tabbed_2" type="radio" /><input id="__tabbed_2_5" name="__tabbed_2" type="radio" /><input id="__tabbed_2_6" name="__tabbed_2" type="radio" /><input id="__tabbed_2_7" name="__tabbed_2" type="radio" /><input id="__tabbed_2_8" name="__tabbed_2" type="radio" /><input id="__tabbed_2_9" name="__tabbed_2" type="radio" /><input id="__tabbed_2_10" name="__tabbed_2" type="radio" /><div class="tabbed-labels"><label for="__tabbed_2_1">Java</label><label for="__tabbed_2_2">C++</label><label for="__tabbed_2_3">Python</label><label for="__tabbed_2_4">Go</label><label for="__tabbed_2_5">JavaScript</label><label for="__tabbed_2_6">TypeScript</label><label for="__tabbed_2_7">C</label><label for="__tabbed_2_8">C#</label><label for="__tabbed_2_9">Swift</label><label for="__tabbed_2_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -1907,7 +1907,7 @@
</div>
</div>
</div>
<p><strong>在列表中添加、插入、删除元素</strong>。相于数组,列表可以自由地添加与删除元素。在列表尾部添加元素的时间复杂度为 <span class="arithmatex">\(O(1)\)</span> ,但插入删除元素的效率仍与数组一样低,时间复杂度为 <span class="arithmatex">\(O(N)\)</span></p>
<p><strong>在列表中添加、插入、删除元素</strong>。相于数组,列表可以自由地添加与删除元素。在列表尾部添加元素的时间复杂度为 <span class="arithmatex">\(O(1)\)</span> ,但插入删除元素的效率仍与数组相同,时间复杂度为 <span class="arithmatex">\(O(N)\)</span></p>
<div class="tabbed-set tabbed-alternate" data-tabs="3:10"><input checked="checked" id="__tabbed_3_1" name="__tabbed_3" type="radio" /><input id="__tabbed_3_2" name="__tabbed_3" type="radio" /><input id="__tabbed_3_3" name="__tabbed_3" type="radio" /><input id="__tabbed_3_4" name="__tabbed_3" type="radio" /><input id="__tabbed_3_5" name="__tabbed_3" type="radio" /><input id="__tabbed_3_6" name="__tabbed_3" type="radio" /><input id="__tabbed_3_7" name="__tabbed_3" type="radio" /><input id="__tabbed_3_8" name="__tabbed_3" type="radio" /><input id="__tabbed_3_9" name="__tabbed_3" type="radio" /><input id="__tabbed_3_10" name="__tabbed_3" type="radio" /><div class="tabbed-labels"><label for="__tabbed_3_1">Java</label><label for="__tabbed_3_2">C++</label><label for="__tabbed_3_3">Python</label><label for="__tabbed_3_4">Go</label><label for="__tabbed_3_5">JavaScript</label><label for="__tabbed_3_6">TypeScript</label><label for="__tabbed_3_7">C</label><label for="__tabbed_3_8">C#</label><label for="__tabbed_3_9">Swift</label><label for="__tabbed_3_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2078,7 +2078,7 @@
</div>
</div>
</div>
<p><strong>遍历列表</strong>。与数组一样,列表可以使用索引遍历,也可以使用 <code>for-each</code> 直接遍历。</p>
<p><strong>遍历列表</strong>。与数组一样,列表可以根据索引遍历,也可以直接遍历各元素</p>
<div class="tabbed-set tabbed-alternate" data-tabs="4:10"><input checked="checked" id="__tabbed_4_1" name="__tabbed_4" type="radio" /><input id="__tabbed_4_2" name="__tabbed_4" type="radio" /><input id="__tabbed_4_3" name="__tabbed_4" type="radio" /><input id="__tabbed_4_4" name="__tabbed_4" type="radio" /><input id="__tabbed_4_5" name="__tabbed_4" type="radio" /><input id="__tabbed_4_6" name="__tabbed_4" type="radio" /><input id="__tabbed_4_7" name="__tabbed_4" type="radio" /><input id="__tabbed_4_8" name="__tabbed_4" type="radio" /><input id="__tabbed_4_9" name="__tabbed_4" type="radio" /><input id="__tabbed_4_10" name="__tabbed_4" type="radio" /><div class="tabbed-labels"><label for="__tabbed_4_1">Java</label><label for="__tabbed_4_2">C++</label><label for="__tabbed_4_3">Python</label><label for="__tabbed_4_4">Go</label><label for="__tabbed_4_5">JavaScript</label><label for="__tabbed_4_6">TypeScript</label><label for="__tabbed_4_7">C</label><label for="__tabbed_4_8">C#</label><label for="__tabbed_4_9">Swift</label><label for="__tabbed_4_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2214,7 +2214,7 @@
</div>
</div>
</div>
<p><strong>拼接两个列表</strong>再创建一个新列表 <code>list1</code> ,我们可以将其中一个列表拼接到另一个的尾部。</p>
<p><strong>拼接两个列表</strong>给定一个新列表 <code>list1</code>,我们可以将列表拼接到原列表的尾部。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="5:10"><input checked="checked" id="__tabbed_5_1" name="__tabbed_5" type="radio" /><input id="__tabbed_5_2" name="__tabbed_5" type="radio" /><input id="__tabbed_5_3" name="__tabbed_5" type="radio" /><input id="__tabbed_5_4" name="__tabbed_5" type="radio" /><input id="__tabbed_5_5" name="__tabbed_5" type="radio" /><input id="__tabbed_5_6" name="__tabbed_5" type="radio" /><input id="__tabbed_5_7" name="__tabbed_5" type="radio" /><input id="__tabbed_5_8" name="__tabbed_5" type="radio" /><input id="__tabbed_5_9" name="__tabbed_5" type="radio" /><input id="__tabbed_5_10" name="__tabbed_5" type="radio" /><div class="tabbed-labels"><label for="__tabbed_5_1">Java</label><label for="__tabbed_5_2">C++</label><label for="__tabbed_5_3">Python</label><label for="__tabbed_5_4">Go</label><label for="__tabbed_5_5">JavaScript</label><label for="__tabbed_5_6">TypeScript</label><label for="__tabbed_5_7">C</label><label for="__tabbed_5_8">C#</label><label for="__tabbed_5_9">Swift</label><label for="__tabbed_5_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2280,7 +2280,7 @@
</div>
</div>
</div>
<p><strong>排序列表</strong>。排序也是常用的方法之一完成列表排序后,我们可以使用在数组类算法题中经常考察的「二分查找」和「双指针」算法</p>
<p><strong>排序列表</strong>。排序也是常用的方法之一完成列表排序后,我们便可以使用在数组类算法题中经常考察的「二分查找」和「双指针」算法。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="6:10"><input checked="checked" id="__tabbed_6_1" name="__tabbed_6" type="radio" /><input id="__tabbed_6_2" name="__tabbed_6" type="radio" /><input id="__tabbed_6_3" name="__tabbed_6" type="radio" /><input id="__tabbed_6_4" name="__tabbed_6" type="radio" /><input id="__tabbed_6_5" name="__tabbed_6" type="radio" /><input id="__tabbed_6_6" name="__tabbed_6" type="radio" /><input id="__tabbed_6_7" name="__tabbed_6" type="radio" /><input id="__tabbed_6_8" name="__tabbed_6" type="radio" /><input id="__tabbed_6_9" name="__tabbed_6" type="radio" /><input id="__tabbed_6_10" name="__tabbed_6" type="radio" /><div class="tabbed-labels"><label for="__tabbed_6_1">Java</label><label for="__tabbed_6_2">C++</label><label for="__tabbed_6_3">Python</label><label for="__tabbed_6_4">Go</label><label for="__tabbed_6_5">JavaScript</label><label for="__tabbed_6_6">TypeScript</label><label for="__tabbed_6_7">C</label><label for="__tabbed_6_8">C#</label><label for="__tabbed_6_9">Swift</label><label for="__tabbed_6_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2334,14 +2334,14 @@
</div>
</div>
</div>
<h2 id="432">4.3.2. &nbsp; 列表简易实现 *<a class="headerlink" href="#432" title="Permanent link">&para;</a></h2>
<p>为了帮助加深对列表的理解,我们在此提供一个列表的简易版本的实现。需要关注三个核心点:</p>
<h2 id="432">4.3.2. &nbsp; 列表实现 *<a class="headerlink" href="#432" title="Permanent link">&para;</a></h2>
<p>为了帮助加深对列表的理解,我们在此提供一个简易版列表实现。需要关注三个核心点:</p>
<ul>
<li><strong>初始容量</strong>:选取一个合理的数组初始容量 <code>initialCapacity</code> 。在本示例中,我们选择 10 作为初始容量。</li>
<li><strong>数量记录</strong>需要声明一个变量 <code>size</code> ,用记录列表当前有多少个元素,并随着元素插入删除实时更新。根据此变量,可以定位列表尾部,以及判断是否需要扩容。</li>
<li><strong>扩容机制</strong>:插入元素可能导致超出列表容量,此时需要扩容列表,方法是建立一个更大的数组来替换当前数组。需要给定一个扩容倍数 <code>extendRatio</code> 在本示例中,我们规定每次将数组扩容至之前的 2 倍。</li>
<li><strong>初始容量</strong>:选取一个合理的数组初始容量。在本示例中,我们选择 10 作为初始容量。</li>
<li><strong>数量记录</strong>:声明一个变量 size,用记录列表当前元素数量,并随着元素插入删除实时更新。根据此变量,我们可以定位列表尾部,以及判断是否需要扩容。</li>
<li><strong>扩容机制</strong>:插入元素可能超出列表容量,此时需要扩容列表。扩容方法是根据扩容倍数创建一个更大的数组,并将当前数组的所有元素依次移动至新数组。在本示例中,我们规定每次将数组扩容至之前的 2 倍。</li>
</ul>
<p>本示例是为了帮助读者对如何实现列表产生直观的认识。实际编程语言中,列表实现远比以下代码复杂且标准,感兴趣的读者可以查阅源码学习。</p>
<p>本示例旨在帮助读者直观理解列表的工作机制。实际编程语言中,列表实现更加标准和复杂,各个参数的设定也非常有考究,例如初始容量、扩容倍数等。感兴趣的读者可以查阅源码进行学习。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="7:10"><input checked="checked" id="__tabbed_7_1" name="__tabbed_7" type="radio" /><input id="__tabbed_7_2" name="__tabbed_7" type="radio" /><input id="__tabbed_7_3" name="__tabbed_7" type="radio" /><input id="__tabbed_7_4" name="__tabbed_7" type="radio" /><input id="__tabbed_7_5" name="__tabbed_7" type="radio" /><input id="__tabbed_7_6" name="__tabbed_7" type="radio" /><input id="__tabbed_7_7" name="__tabbed_7" type="radio" /><input id="__tabbed_7_8" name="__tabbed_7" type="radio" /><input id="__tabbed_7_9" name="__tabbed_7" type="radio" /><input id="__tabbed_7_10" name="__tabbed_7" type="radio" /><div class="tabbed-labels"><label for="__tabbed_7_1">Java</label><label for="__tabbed_7_2">C++</label><label for="__tabbed_7_3">Python</label><label for="__tabbed_7_4">Go</label><label for="__tabbed_7_5">JavaScript</label><label for="__tabbed_7_6">TypeScript</label><label for="__tabbed_7_7">C</label><label for="__tabbed_7_8">C#</label><label for="__tabbed_7_9">Swift</label><label for="__tabbed_7_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">

View File

@ -1681,11 +1681,11 @@
<h1 id="44">4.4. &nbsp; 小结<a class="headerlink" href="#44" title="Permanent link">&para;</a></h1>
<ul>
<li>数组和链表是两种基本数据结构,代表数据在计算机内存中的两种存储方式,即连续空间存储和离散空间存储。两者的优点与缺点呈现出此消彼长的关系</li>
<li>数组支持随机访问、内存空间占用小;但插入删除元素效率低,且初始化后长度不可变。</li>
<li>链表通过更改指针实现高效的结点插入与删除,且可以灵活地修改长度;但结点访问效率低、占用内存多。常见的链表类型单向链表、循环链表、双向链表。</li>
<li>列表又称动态数组,是基于数组实现的一种数据结构,其保存了数组的优势,可以灵活改变长度。列表的出现大大提升了数组的用性,但副作用是会造成部分内存空间浪费。</li>
<li>下表总结对比了数组与链表的各项特性。</li>
<li>数组和链表是两种基本数据结构,分别代表数据在计算机内存中的连续空间存储和离散空间存储方式。两者的优缺点呈现出互补的特性</li>
<li>数组支持随机访问、占用内存较少;但插入删除元素效率低,且初始化后长度不可变。</li>
<li>链表通过更改指针实现高效的结点插入与删除,且可以灵活调整长度;但结点访问效率低、占用内存多。常见的链表类型包括单向链表、循环链表、双向链表。</li>
<li>动态数组,又称列表,是基于数组实现的一种数据结构。它保留了数组的优势,同时可以灵活调整长度。列表的出现极大地提高了数组的用性,但可能导致部分内存空间浪费。</li>
<li>下表总结对比了数组与链表的各项特性。</li>
</ul>
<div class="center-table">
<table>
@ -1720,12 +1720,12 @@
</tbody>
</table>
</div>
<div class="admonition question">
<p class="admonition-title">缓存局部性的简单解释</p>
<div class="admonition note">
<p class="admonition-title">缓存局部性</p>
<p>在计算机中,数据读写速度排序是“硬盘 &lt; 内存 &lt; CPU 缓存”。当我们访问数组元素时,计算机不仅会加载它,还会缓存其周围的其它数据,从而借助高速缓存来提升后续操作的执行速度。链表则不然,计算机只能挨个地缓存各个结点,这样的多次“搬运”降低了整体效率。</p>
</div>
<ul>
<li>下表对比了数组与链表各种操作效率。</li>
<li>下表对比了数组与链表各种操作上的效率。</li>
</ul>
<div class="center-table">
<table>

View File

@ -1795,35 +1795,35 @@
<h1 id="21">2.1. &nbsp; 算法效率评估<a class="headerlink" href="#21" title="Permanent link">&para;</a></h1>
<h2 id="211">2.1.1. &nbsp; 算法评价维度<a class="headerlink" href="#211" title="Permanent link">&para;</a></h2>
<p>在开始学习算法之前,我们首先要想清楚算法的设计目标是什么,或者说,如何评判算法的好与坏。整体上看,我们设计算法时追求两个层面的目标</p>
<p>在开始学习算法之前,我们首先需要明确算法的设计目标,换句话说,我们应该如何评判算法的优劣。从总体上看,算法设计追求以下两个层面的目标</p>
<ol>
<li><strong>找到问题解法</strong>。算法需要能够在规定的输入范围,可靠地求得问题的正确解。</li>
<li><strong>寻求最优解法</strong>。同一个问题可能存在多种解法,我们希望算法效率尽可能的高</li>
<li><strong>找到问题解法</strong>。算法需要在规定的输入范围,可靠地求得问题的正确解。</li>
<li><strong>寻求最优解法</strong>。同一个问题可能存在多种解法,我们希望找到尽可能高效的算法。</li>
</ol>
<p>换言之,在可以解决问题的前提下,算法效率则是主要评价维度,包括:</p>
<p>因此,在能够解决问题的前提下,算法效率成为主要评价维度,主要包括:</p>
<ul>
<li><strong>时间效率</strong>,即算法运行速度的快慢。</li>
<li><strong>空间效率</strong>,即算法占用内存空间大小。</li>
<li><strong>时间效率</strong>,即算法运行速度的快慢。</li>
<li><strong>空间效率</strong>,即算法占用内存空间大小。</li>
</ul>
<p>数据结构与算法追求“运行速度快、占用内存少”,而如何去评价算法效率则是非常重要的问题,因为只有知道如何评价法,才能去做算法间的对比分析,以及优化算法设计。</p>
<p>数据结构与算法的终极目标是“又快又省”。了解如何评估算法效率非常重要,因为只有掌握了评价法,我们才能进行算法间的对比分析,从而指导算法设计与优化</p>
<h2 id="212">2.1.2. &nbsp; 效率评估方法<a class="headerlink" href="#212" title="Permanent link">&para;</a></h2>
<h3 id="_1">实际测试<a class="headerlink" href="#_1" title="Permanent link">&para;</a></h3>
<p>假设我们现在有算法 A 和 算法 B ,都能解决同一问题,现在需要对比两个算法之间的效率。我们能够想到的最直接的方式,就是找一台计算机,两个算法都完整跑一遍,并监控记录运行时间和内存占用情况。这种评估方式能够反映真实情况,但也存在很大的硬伤</p>
<p><strong>难以排除测试环境的干扰因素</strong>。硬件配置会影响算法的性能表现。例如,在某台计算机中,算法 A 比算法 B 运行时间更短;但换到另一台配置不同的计算机中,可能得到相反的测试结果。这意味着我们需要在各种机器上展开测试,而这是不现实的。</p>
<p><strong>展开完整测试非常耗费资源</strong>。随着输入数据量的大小变化,算法会现出不同的效率表现。比如,有可能输入数据量较小时,算法 A 运行时间短于算法 B ,而在输入数据量较大时,测试结果截然相反。因此,若想要达到具有说服力的对比结果,那么需要输入各种体量数据,这样的测试需要占用大量计算资源。</p>
<p>假设我们现在有算法 A 和算法 B它们都能解决同一问题,现在需要对比两个算法的效率。我们最直接的方就是找一台计算机,运行这两个算法,并监控记录它们的运行时间和内存占用情况。这种评估方式能够反映真实情况,但也存在较大局限性</p>
<p><strong>难以排除测试环境的干扰因素</strong>。硬件配置会影响算法的性能表现。例如,在某台计算机中,算法 A 的运行时间比算法 B 短;但另一台配置不同的计算机中,我们可能得到相反的测试结果。这意味着我们需要在各种机器上进行测试,而这是不现实的。</p>
<p><strong>展开完整测试非常耗费资源</strong>。随着输入数据量的变化,算法会现出不同的效率。例如,输入数据量较小时,算法 A 运行时间可能短于算法 B;而输入数据量较大时,测试结果可能相反。因此,为了得到有说服力的结论,我们需要测试各种规模的输入数据,这样需要占用大量计算资源。</p>
<h3 id="_2">理论估算<a class="headerlink" href="#_2" title="Permanent link">&para;</a></h3>
<p>既然实际测试具有大的局限性,那么我们是否可以仅通过一些计算,就获知算法的效率水平呢?答案是肯定的,我们将此估算方法称为「复杂度分析 Complexity Analysis」或「渐近复杂度分析 Asymptotic Complexity Analysis」。</p>
<p><strong>复杂度分析评估的是算法运行效率随着输入数据量增多时的增长趋势</strong>。这句话有些拗口,我们可以将其分为三个重点来理解:</p>
<p>由于实际测试具有大的局限性,我们可以考虑仅通过一些计算来评估算法的效率。这种估算方法称为「复杂度分析 Complexity Analysis」或「渐近复杂度分析 Asymptotic Complexity Analysis」。</p>
<p><strong>复杂度分析评估的是算法运行效率随着输入数据量增多时的增长趋势</strong>。这个定义有些拗口,我们可以将其分为三个重点来理解:</p>
<ul>
<li>“算法运行效率”可分为“运行时间”和“占用空间”,进而可将复杂度分为「时间复杂度 Time Complexity」和「空间复杂度 Space Complexity」</li>
<li>“随着输入数据量增多时”表复杂度与输入数据量有关,反映算法运行效率与输入数据量之间的关系;</li>
<li>“增长趋势”表示复杂度分析不关心算法具体使用了多少时间或占用了多少空间,而是给出一种“趋势性分析”</li>
<li>“算法运行效率”可分为“运行时间”和“占用空间”,因此我们可以将复杂度分为「时间复杂度 Time Complexity」和「空间复杂度 Space Complexity」</li>
<li>“随着输入数据量增多时”表复杂度与输入数据量有关,反映算法运行效率与输入数据量之间的关系;</li>
<li>“增长趋势”表示复杂度分析关注的是算法时间与空间的增长趋势,而非具体的运行时间或占用空间</li>
</ul>
<p><strong>复杂度分析克服了实际测试方法的弊端</strong>一是独立于测试环境,分析结果适用于所有运行平台。二是可以体现不同数据量下的算法效率,尤其是可以反映大数据量下的算法性能。</p>
<p>如果感觉对复杂度分析的概念一知半解,无需担心,后续章节会展开介绍。</p>
<p><strong>复杂度分析克服了实际测试方法的弊端</strong>首先,它独立于测试环境,因此分析结果适用于所有运行平台。其次,它可以体现不同数据量下的算法效率,尤其是大数据量下的算法性能。</p>
<p>如果对复杂度分析的概念仍感到困惑,无需担心,我们会在后续章节详细介绍。</p>
<h2 id="213">2.1.3. &nbsp; 复杂度分析重要性<a class="headerlink" href="#213" title="Permanent link">&para;</a></h2>
<p>复杂度分析给出一把评算法效率的“标尺”,告诉我们执行某个算法需要多少时间和空间资源,也让我们可以开展不同算法之间的效率对比</p>
<p>复杂度是个数学概念,对于初学者可能比较抽象,学习难度相对较高。从这个角度出发,其并不适合作为第一章内容。但是,当我们讨论某个数据结构或算法的特点时,难以避免要分析它的运行速度和空间使用情况。<strong>因此,在展开学习数据结构与算法之前,建议读者先对复杂度建立初步的了解,并能够完成简单案例的复杂度分析</strong></p>
<p>复杂度分析为我们提供了一把评算法效率的“标尺”,告诉我们执行某个算法所需的时间和空间资源,并使我们能够对比不同算法之间的效率。</p>
<p>复杂度是个数学概念,对于初学者可能比较抽象,学习难度相对较高。从这个角度看,复杂度分析可能不太适合作为第一章内容。然而,当我们讨论某个数据结构或算法的特点时,我们难以避免要分析运行速度和空间使用情况。<strong>因此,在深入学习数据结构与算法之前,建议读者先对复杂度建立初步的了解,并能够完成简单案例的复杂度分析</strong></p>

View File

@ -1836,24 +1836,22 @@
<h1 id="23">2.3. &nbsp; 空间复杂度<a class="headerlink" href="#23" title="Permanent link">&para;</a></h1>
<p>「空间复杂度 Space Complexity」统计 <strong>算法使用内存空间随着数据量变大时的增长趋势</strong>。这个概念与时间复杂度类似。</p>
<p>「空间复杂度 Space Complexity」用于衡量算法使用内存空间随着数据量变大时的增长趋势。这个概念与时间复杂度非常类似。</p>
<h2 id="231">2.3.1. &nbsp; 算法相关空间<a class="headerlink" href="#231" title="Permanent link">&para;</a></h2>
<p>算法运行中使用的内存空间主要以下几种:</p>
<p>算法运行过程中使用的内存空间主要包括以下几种:</p>
<ul>
<li>「输入空间」用于存储算法的输入数据;</li>
<li>「暂存空间」用于存储算法运行中的变量、对象、函数上下文等数据;</li>
<li>「暂存空间」用于存储算法运行过程中的变量、对象、函数上下文等数据;</li>
<li>「输出空间」用于存储算法的输出数据;</li>
</ul>
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p>通常情况下,空间复杂度统计范围是「暂存空间」+「输出空间」。</p>
</div>
<p>暂存空间可分为三个部分:</p>
<p>暂存空间可以进一步划分为三个部分:</p>
<ul>
<li>「暂存数据」用于保存算法运行中的各种 <strong>常量、变量、对象</strong> 等。</li>
<li>「栈帧空间」用于保存调用函数的上下文数据。系统每次调用函数都会在栈顶部创建一个栈帧,函数返回,栈帧空间会被释放。</li>
<li>「指令空间」用于保存编译后的程序指令,<strong>在实际统计中一般忽略不计</strong></li>
<li>「暂存数据」用于保存算法运行过程中的各种常量、变量、对象等。</li>
<li>「栈帧空间」用于保存调用函数的上下文数据。系统每次调用函数都会在栈顶部创建一个栈帧,函数返回,栈帧空间会被释放。</li>
<li>「指令空间」用于保存编译后的程序指令,在实际统计中通常忽略不计。</li>
</ul>
<p>因此,在分析一段程序的空间复杂度时,我们一般统计 <strong>暂存数据、输出数据、栈帧空间</strong> 三部分。</p>
<p><img alt="算法使用的相关空间" src="../space_complexity.assets/space_types.png" /></p>
<p align="center"> Fig. 算法使用的相关空间 </p>
@ -2067,11 +2065,11 @@
</div>
</div>
<h2 id="232">2.3.2. &nbsp; 推算方法<a class="headerlink" href="#232" title="Permanent link">&para;</a></h2>
<p>空间复杂度的推算方法时间复杂度总体类似,只是统计“计算操作数量”变为统计“使用空间大小”。与时间复杂度不同的是,<strong>我们一般只关注「最差空间复杂度」</strong>这是因为内存空间是一硬性要求,我们必须保在所有输入数据下都有足够的内存空间预留。</p>
<p><strong>最差空间复杂度中的“最差”有两层含义</strong>,分别输入数据的最差分布算法运行中的最差时间点。</p>
<p>空间复杂度的推算方法时间复杂度大致相同,只是统计对象从“计算操作数量”转为“使用空间大小”。与时间复杂度不同的是,<strong>我们通常只关注「最差空间复杂度」</strong>这是因为内存空间是一硬性要求,我们必须保在所有输入数据下都有足够的内存空间预留。</p>
<p><strong>最差空间复杂度中的“最差”有两层含义</strong>,分别输入数据的最差分布算法运行过程中的最差时间点。</p>
<ul>
<li><strong>以最差输入数据为准</strong>。当 <span class="arithmatex">\(n &lt; 10\)</span> 时,空间复杂度为 <span class="arithmatex">\(O(1)\)</span> ;但<span class="arithmatex">\(n &gt; 10\)</span> 时,初始化的数组 <code>nums</code> 使<span class="arithmatex">\(O(n)\)</span> 空间;因此最差空间复杂度为 <span class="arithmatex">\(O(n)\)</span> </li>
<li><strong>以算法运行过程中的峰值内存为准</strong>。程序在执行最后一行之前,使<span class="arithmatex">\(O(1)\)</span> 空间;当初始化数组 <code>nums</code> 时,程序使<span class="arithmatex">\(O(n)\)</span> 空间;因此最差空间复杂度为 <span class="arithmatex">\(O(n)\)</span> </li>
<li><strong>以最差输入数据为准</strong>。当 <span class="arithmatex">\(n &lt; 10\)</span> 时,空间复杂度为 <span class="arithmatex">\(O(1)\)</span> ;但当 <span class="arithmatex">\(n &gt; 10\)</span> 时,初始化的数组 <code>nums</code> <span class="arithmatex">\(O(n)\)</span> 空间;因此最差空间复杂度为 <span class="arithmatex">\(O(n)\)</span> </li>
<li><strong>以算法运行过程中的峰值内存为准</strong>例如,程序在执行最后一行之前,<span class="arithmatex">\(O(1)\)</span> 空间;当初始化数组 <code>nums</code> 时,程序<span class="arithmatex">\(O(n)\)</span> 空间;因此最差空间复杂度为 <span class="arithmatex">\(O(n)\)</span> </li>
</ul>
<div class="tabbed-set tabbed-alternate" data-tabs="2:10"><input checked="checked" id="__tabbed_2_1" name="__tabbed_2" type="radio" /><input id="__tabbed_2_2" name="__tabbed_2" type="radio" /><input id="__tabbed_2_3" name="__tabbed_2" type="radio" /><input id="__tabbed_2_4" name="__tabbed_2" type="radio" /><input id="__tabbed_2_5" name="__tabbed_2" type="radio" /><input id="__tabbed_2_6" name="__tabbed_2" type="radio" /><input id="__tabbed_2_7" name="__tabbed_2" type="radio" /><input id="__tabbed_2_8" name="__tabbed_2" type="radio" /><input id="__tabbed_2_9" name="__tabbed_2" type="radio" /><input id="__tabbed_2_10" name="__tabbed_2" type="radio" /><div class="tabbed-labels"><label for="__tabbed_2_1">Java</label><label for="__tabbed_2_2">C++</label><label for="__tabbed_2_3">Python</label><label for="__tabbed_2_4">Go</label><label for="__tabbed_2_5">JavaScript</label><label for="__tabbed_2_6">TypeScript</label><label for="__tabbed_2_7">C</label><label for="__tabbed_2_8">C#</label><label for="__tabbed_2_9">Swift</label><label for="__tabbed_2_10">Zig</label></div>
<div class="tabbed-content">
@ -2165,7 +2163,7 @@
</div>
</div>
</div>
<p><strong>在递归函数中,需要注意统计栈帧空间</strong>。例如函数 <code>loop()</code>在循环中调用了 <span class="arithmatex">\(n\)</span><code>function()</code> ,每轮中的 <code>function()</code> 都返回并释放了栈帧空间,因此空间复杂度仍为 <span class="arithmatex">\(O(1)\)</span> 。而递归函数 <code>recur()</code> 在运行中会同时存在 <span class="arithmatex">\(n\)</span> 个未返回的 <code>recur()</code> ,从而使<span class="arithmatex">\(O(n)\)</span> 的栈帧空间。</p>
<p><strong>在递归函数中,需要注意统计栈帧空间</strong>。例如函数 <code>loop()</code> 在循环中调用了 <span class="arithmatex">\(n\)</span><code>function()</code>,每轮中的 <code>function()</code> 都返回并释放了栈帧空间,因此空间复杂度仍为 <span class="arithmatex">\(O(1)\)</span> 。而递归函数 <code>recur()</code> 在运行过程中会同时存在 <span class="arithmatex">\(n\)</span> 个未返回的 <code>recur()</code>,从而<span class="arithmatex">\(O(n)\)</span> 的栈帧空间。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="3:10"><input checked="checked" id="__tabbed_3_1" name="__tabbed_3" type="radio" /><input id="__tabbed_3_2" name="__tabbed_3" type="radio" /><input id="__tabbed_3_3" name="__tabbed_3" type="radio" /><input id="__tabbed_3_4" name="__tabbed_3" type="radio" /><input id="__tabbed_3_5" name="__tabbed_3" type="radio" /><input id="__tabbed_3_6" name="__tabbed_3" type="radio" /><input id="__tabbed_3_7" name="__tabbed_3" type="radio" /><input id="__tabbed_3_8" name="__tabbed_3" type="radio" /><input id="__tabbed_3_9" name="__tabbed_3" type="radio" /><input id="__tabbed_3_10" name="__tabbed_3" type="radio" /><div class="tabbed-labels"><label for="__tabbed_3_1">Java</label><label for="__tabbed_3_2">C++</label><label for="__tabbed_3_3">Python</label><label for="__tabbed_3_4">Go</label><label for="__tabbed_3_5">JavaScript</label><label for="__tabbed_3_6">TypeScript</label><label for="__tabbed_3_7">C</label><label for="__tabbed_3_8">C#</label><label for="__tabbed_3_9">Swift</label><label for="__tabbed_3_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2346,7 +2344,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n^2) &lt; O(2^n) \newline
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p>部分示例代码需要一些前置知识,包括数组、链表、二叉树、递归算法等。如果遇到看不懂的地方无需担心,可以在学习完后面章节后再来复习,现阶段先聚焦在理解空间复杂度含义和推算方法</p>
<p>部分示例代码需要一些前置知识,包括数组、链表、二叉树、递归算法等。如果遇到看不懂的地方无需担心,可以在学习完后面章节后再来复习,现阶段我们先专注于理解空间复杂度含义和推算方法。</p>
</div>
<h3 id="o1">常数阶 <span class="arithmatex">\(O(1)\)</span><a class="headerlink" href="#o1" title="Permanent link">&para;</a></h3>
<p>常数阶常见于数量与输入数据大小 <span class="arithmatex">\(n\)</span> 无关的常量、变量、对象。</p>
@ -2805,7 +2803,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n^2) &lt; O(2^n) \newline
<p align="center"> Fig. 递归函数产生的线性阶空间复杂度 </p>
<h3 id="on2">平方阶 <span class="arithmatex">\(O(n^2)\)</span><a class="headerlink" href="#on2" title="Permanent link">&para;</a></h3>
<p>平方阶常见于元素数量与 <span class="arithmatex">\(n\)</span> 成平方关系的矩阵、图</p>
<p>平方阶常见于矩阵和图,元素数量与 <span class="arithmatex">\(n\)</span> 成平方关系。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="7:10"><input checked="checked" id="__tabbed_7_1" name="__tabbed_7" type="radio" /><input id="__tabbed_7_2" name="__tabbed_7" type="radio" /><input id="__tabbed_7_3" name="__tabbed_7" type="radio" /><input id="__tabbed_7_4" name="__tabbed_7" type="radio" /><input id="__tabbed_7_5" name="__tabbed_7" type="radio" /><input id="__tabbed_7_6" name="__tabbed_7" type="radio" /><input id="__tabbed_7_7" name="__tabbed_7" type="radio" /><input id="__tabbed_7_8" name="__tabbed_7" type="radio" /><input id="__tabbed_7_9" name="__tabbed_7" type="radio" /><input id="__tabbed_7_10" name="__tabbed_7" type="radio" /><div class="tabbed-labels"><label for="__tabbed_7_1">Java</label><label for="__tabbed_7_2">C++</label><label for="__tabbed_7_3">Python</label><label for="__tabbed_7_4">Go</label><label for="__tabbed_7_5">JavaScript</label><label for="__tabbed_7_6">TypeScript</label><label for="__tabbed_7_7">C</label><label for="__tabbed_7_8">C#</label><label for="__tabbed_7_9">Swift</label><label for="__tabbed_7_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2947,7 +2945,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n^2) &lt; O(2^n) \newline
</div>
</div>
</div>
<p>在以下递归函数中,同时存在 <span class="arithmatex">\(n\)</span> 个未返回的 <code>algorithm()</code> ,并且每个函数中都初始化了一个数组,长度分别为 <span class="arithmatex">\(n, n-1, n-2, ..., 2, 1\)</span> ,平均长度为 <span class="arithmatex">\(\frac{n}{2}\)</span> ,因此总体使<span class="arithmatex">\(O(n^2)\)</span> 空间。</p>
<p>在以下递归函数中,同时存在 <span class="arithmatex">\(n\)</span> 个未返回的 <code>algorithm()</code>,并且每个函数中都初始化了一个数组,长度分别为 <span class="arithmatex">\(n, n-1, n-2, ..., 2, 1\)</span> ,平均长度为 <span class="arithmatex">\(\frac{n}{2}\)</span> ,因此总体<span class="arithmatex">\(O(n^2)\)</span> 空间。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="8:10"><input checked="checked" id="__tabbed_8_1" name="__tabbed_8" type="radio" /><input id="__tabbed_8_2" name="__tabbed_8" type="radio" /><input id="__tabbed_8_3" name="__tabbed_8" type="radio" /><input id="__tabbed_8_4" name="__tabbed_8" type="radio" /><input id="__tabbed_8_5" name="__tabbed_8" type="radio" /><input id="__tabbed_8_6" name="__tabbed_8" type="radio" /><input id="__tabbed_8_7" name="__tabbed_8" type="radio" /><input id="__tabbed_8_8" name="__tabbed_8" type="radio" /><input id="__tabbed_8_9" name="__tabbed_8" type="radio" /><input id="__tabbed_8_10" name="__tabbed_8" type="radio" /><div class="tabbed-labels"><label for="__tabbed_8_1">Java</label><label for="__tabbed_8_2">C++</label><label for="__tabbed_8_3">Python</label><label for="__tabbed_8_4">Go</label><label for="__tabbed_8_5">JavaScript</label><label for="__tabbed_8_6">TypeScript</label><label for="__tabbed_8_7">C</label><label for="__tabbed_8_8">C#</label><label for="__tabbed_8_9">Swift</label><label for="__tabbed_8_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -3057,7 +3055,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n^2) &lt; O(2^n) \newline
<p align="center"> Fig. 递归函数产生的平方阶空间复杂度 </p>
<h3 id="o2n">指数阶 <span class="arithmatex">\(O(2^n)\)</span><a class="headerlink" href="#o2n" title="Permanent link">&para;</a></h3>
<p>指数阶常见于二叉树。高度为 <span class="arithmatex">\(n\)</span> 的「满二叉树」的点数量为 <span class="arithmatex">\(2^n - 1\)</span> 使<span class="arithmatex">\(O(2^n)\)</span> 空间。</p>
<p>指数阶常见于二叉树。高度为 <span class="arithmatex">\(n\)</span> 的「满二叉树」的点数量为 <span class="arithmatex">\(2^n - 1\)</span> <span class="arithmatex">\(O(2^n)\)</span> 空间。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="9:10"><input checked="checked" id="__tabbed_9_1" name="__tabbed_9" type="radio" /><input id="__tabbed_9_2" name="__tabbed_9" type="radio" /><input id="__tabbed_9_3" name="__tabbed_9" type="radio" /><input id="__tabbed_9_4" name="__tabbed_9" type="radio" /><input id="__tabbed_9_5" name="__tabbed_9" type="radio" /><input id="__tabbed_9_6" name="__tabbed_9" type="radio" /><input id="__tabbed_9_7" name="__tabbed_9" type="radio" /><input id="__tabbed_9_8" name="__tabbed_9" type="radio" /><input id="__tabbed_9_9" name="__tabbed_9" type="radio" /><input id="__tabbed_9_10" name="__tabbed_9" type="radio" /><div class="tabbed-labels"><label for="__tabbed_9_1">Java</label><label for="__tabbed_9_2">C++</label><label for="__tabbed_9_3">Python</label><label for="__tabbed_9_4">Go</label><label for="__tabbed_9_5">JavaScript</label><label for="__tabbed_9_6">TypeScript</label><label for="__tabbed_9_7">C</label><label for="__tabbed_9_8">C#</label><label for="__tabbed_9_9">Swift</label><label for="__tabbed_9_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -3174,9 +3172,9 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n^2) &lt; O(2^n) \newline
<p align="center"> Fig. 满二叉树产生的指数阶空间复杂度 </p>
<h3 id="olog-n">对数阶 <span class="arithmatex">\(O(\log n)\)</span><a class="headerlink" href="#olog-n" title="Permanent link">&para;</a></h3>
<p>对数阶常见于分治算法数据类型转换等。</p>
<p>例如归并排序」,长度为 <span class="arithmatex">\(n\)</span> 的数组可以形成高度为 <span class="arithmatex">\(\log n\)</span> 的递归树,因此空间复杂度为 <span class="arithmatex">\(O(\log n)\)</span></p>
<p>再例如数字转化为字符串,输入任意正整数 <span class="arithmatex">\(n\)</span> ,它的位数为 <span class="arithmatex">\(\log_{10} n\)</span> ,即对应字符串长度为 <span class="arithmatex">\(\log_{10} n\)</span> ,因此空间复杂度为 <span class="arithmatex">\(O(\log_{10} n) = O(\log n)\)</span></p>
<p>对数阶常见于分治算法数据类型转换等。</p>
<p>例如归并排序”算法,输入长度为 <span class="arithmatex">\(n\)</span> 的数组,每轮递归将数组从中点划分为两半,形成高度为 <span class="arithmatex">\(\log n\)</span> 的递归树,使用 <span class="arithmatex">\(O(\log n)\)</span> 栈帧空间</p>
<p>再例如数字转化为字符串,输入任意正整数 <span class="arithmatex">\(n\)</span> ,它的位数为 <span class="arithmatex">\(\log_{10} n\)</span> ,即对应字符串长度为 <span class="arithmatex">\(\log_{10} n\)</span> ,因此空间复杂度为 <span class="arithmatex">\(O(\log_{10} n) = O(\log n)\)</span></p>

View File

@ -1766,9 +1766,9 @@
<h1 id="24">2.4. &nbsp; 权衡时间与空间<a class="headerlink" href="#24" title="Permanent link">&para;</a></h1>
<p>理想情况下,我们希望算法的时间复杂度和空间复杂度都能达到最优,而实际上,同时优化时间复杂度和空间复杂度是非常困难的。</p>
<p><strong>降低时间复杂度,往往是以提升空间复杂度为代价,反之亦然</strong>。我们牺牲内存空间来提升算法运行速度的思路称为「以空间换时间」;反之,称之为「以时间换空间」。选择哪种思路取决于我们更看重哪个方面。</p>
<p>大多数情况下,时间都是比空间更宝贵的,只要空间复杂度不要太离谱、能接受就行,<strong>因此以空间换时间最为常用</strong></p>
<p>理想情况下,我们希望算法的时间复杂度和空间复杂度都能达到最优。然而在实际情况中,同时优化时间复杂度和空间复杂度通常是非常困难的。</p>
<p><strong>降低时间复杂度通常需要以提升空间复杂度为代价,反之亦然</strong>。我们牺牲内存空间来提升算法运行速度的思路称为「以空间换时间」;反之,称之为「以时间换空间」。</p>
<p>选择哪种思路取决于我们更看重哪个方面。在大多数情况下,时间比空间更宝贵,因此以空间换时间通常是更常用的策略。当然,在数据量很大的情况下,控制空间复杂度也是非常重要的</p>
<h2 id="241">2.4.1. &nbsp; 示例题目 *<a class="headerlink" href="#241" title="Permanent link">&para;</a></h2>
<p>以 LeetCode 全站第一题 <a href="https://leetcode.cn/problems/two-sum/">两数之和</a> 为例。</p>
<div class="admonition question">
@ -1777,9 +1777,9 @@
<p>你可以假设每种输入只会对应一个答案。但是,数组中同一个元素在答案里不能重复出现。</p>
<p>你可以按任意顺序返回答案。</p>
</div>
<p>「暴力枚举」和「辅助哈希表」分别对应 <strong>空间最优</strong><strong>时间最优</strong> 的两种解法。本着时间比空间更宝贵的原则,后者是本题的最佳解法。</p>
<p>「暴力枚举」和「辅助哈希表」分别对应“空间最优”和“时间最优”的两种解法。遵循时间比空间更宝贵的原则,后者是本题的最佳解法。</p>
<h3 id="_1">方法一:暴力枚举<a class="headerlink" href="#_1" title="Permanent link">&para;</a></h3>
<p>考虑直接遍历所有所有可能性。通过开启一个两层循环,判断两个整数的和是否为 <code>target</code> ,若是则返回它的索引(即下标)即可</p>
<p>考虑直接遍历所有可能的组合。通过开启一个两层循环,判断两个整数的和是否为 <code>target</code> ,若是则返回它的索引(即下标)。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="1:10"><input checked="checked" id="__tabbed_1_1" name="__tabbed_1" type="radio" /><input id="__tabbed_1_2" name="__tabbed_1" type="radio" /><input id="__tabbed_1_3" name="__tabbed_1" type="radio" /><input id="__tabbed_1_4" name="__tabbed_1" type="radio" /><input id="__tabbed_1_5" name="__tabbed_1" type="radio" /><input id="__tabbed_1_6" name="__tabbed_1" type="radio" /><input id="__tabbed_1_7" name="__tabbed_1" type="radio" /><input id="__tabbed_1_8" name="__tabbed_1" type="radio" /><input id="__tabbed_1_9" name="__tabbed_1" type="radio" /><input id="__tabbed_1_10" name="__tabbed_1" type="radio" /><div class="tabbed-labels"><label for="__tabbed_1_1">Java</label><label for="__tabbed_1_2">C++</label><label for="__tabbed_1_3">Python</label><label for="__tabbed_1_4">Go</label><label for="__tabbed_1_5">JavaScript</label><label for="__tabbed_1_6">TypeScript</label><label for="__tabbed_1_7">C</label><label for="__tabbed_1_8">C#</label><label for="__tabbed_1_9">Swift</label><label for="__tabbed_1_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -1928,9 +1928,9 @@
</div>
</div>
</div>
<p>该方法的时间复杂度为 <span class="arithmatex">\(O(N^2)\)</span> ,空间复杂度为 <span class="arithmatex">\(O(1)\)</span> <strong>属于时间换空间</strong>方法时间复杂度高,在大数据量下非常耗时。</p>
<p>该方法的时间复杂度为 <span class="arithmatex">\(O(n^2)\)</span> ,空间复杂度为 <span class="arithmatex">\(O(1)\)</span> <strong>属于时间换空间</strong>方法时间复杂度高,在大数据量下非常耗时。</p>
<h3 id="_2">方法二:辅助哈希表<a class="headerlink" href="#_2" title="Permanent link">&para;</a></h3>
<p>考虑借助一个哈希表key 为数组元素、value 为元素索引。循环遍历数组中的每个元素 <code>num</code> ,并执行:</p>
<p>考虑借助一个哈希表key-value 分别为数组元素和元素索引。循环遍历数组中的每个元素 num,并执行:</p>
<ol>
<li>判断数字 <code>target - num</code> 是否在哈希表中,若是则直接返回该两个元素的索引;</li>
<li>将元素 <code>num</code> 和其索引添加进哈希表;</li>
@ -2096,7 +2096,7 @@
</div>
</div>
</div>
<p>该方法的时间复杂度为 <span class="arithmatex">\(O(N)\)</span> ,空间复杂度为 <span class="arithmatex">\(O(N)\)</span> <strong>体现空间换时间</strong>本方法虽然引入了额外空间使用,但时间和空间使用整体更加均衡,因此本题最优解法。</p>
<p>该方法的时间复杂度为 <span class="arithmatex">\(O(N)\)</span> ,空间复杂度为 <span class="arithmatex">\(O(N)\)</span> <strong>体现了以空间换时间</strong>尽管此方法引入了额外空间使用,但时间和空间的整体效率更为均衡,因此它是本题最优解法。</p>

View File

@ -1756,25 +1756,25 @@
<h1 id="25">2.5. &nbsp; 小结<a class="headerlink" href="#25" title="Permanent link">&para;</a></h1>
<h3 id="_1">算法效率评估<a class="headerlink" href="#_1" title="Permanent link">&para;</a></h3>
<ul>
<li>时间效率和空间效率是算法性能的两个重要的评价维度。</li>
<li>我们可以通过实际测试来评估算法效率,但难以除测试环境的干扰,并且非常耗费计算资源。</li>
<li>复杂度分析克服实际测试的弊端,分析结果适用于所有运行平台,并且可以体现不同数据大小下的算法效率。</li>
<li>时间效率和空间效率是评价算法性能的两个关键维度。</li>
<li>我们可以通过实际测试来评估算法效率,但难以除测试环境的影响,且会耗费大量计算资源。</li>
<li>复杂度分析可以克服实际测试的弊端,分析结果适用于所有运行平台,并且能够揭示算法在不同数据规模下的效率。</li>
</ul>
<h3 id="_2">时间复杂度<a class="headerlink" href="#_2" title="Permanent link">&para;</a></h3>
<ul>
<li>时间复杂度统计算法运行时间随数据量变大时的增长趋势,可以有效评估算法效率,但在某些情况下可能失效,如在输入数据量较小或时间复杂度相同时,无法精确对比算法效率的优劣</li>
<li>最差时间复杂度使用大 <span class="arithmatex">\(O\)</span> 符号表示,即函数渐上界,反映当 <span class="arithmatex">\(n\)</span>正无穷时,<span class="arithmatex">\(T(n)\)</span> 处于何种增长级别。</li>
<li>推算时间复杂度分为两步,首先统计计算操作数量,判断渐上界。</li>
<li>常见时间复杂度从小到大排列有 <span class="arithmatex">\(O(1)\)</span> , <span class="arithmatex">\(O(\log n)\)</span> , <span class="arithmatex">\(O(n)\)</span> , <span class="arithmatex">\(O(n \log n)\)</span> , <span class="arithmatex">\(O(n^2)\)</span> , <span class="arithmatex">\(O(2^n)\)</span> , <span class="arithmatex">\(O(n!)\)</span></li>
<li>某些算法的时间复杂度不是恒定的,而是与输入数据的分布有关。时间复杂度分为最差时间复杂度最佳时间复杂度,后者几乎不用,因为输入数据需要满足苛刻的条件才能达到最佳情况。</li>
<li>平均时间复杂度可以反映在随机数据输入下的算法效率,最贴合实际使用情况下的算法性能。计算平均时间复杂度需要统计输入数据分布以及综合后的数学期望。</li>
<li>时间复杂度用于衡量算法运行时间随数据量增长趋势,可以有效评估算法效率,但在某些情况下可能失效,如在输入数据量较小或时间复杂度相同时,无法精确对比算法效率的优劣。</li>
<li>最差时间复杂度使用大 <span class="arithmatex">\(O\)</span> 符号表示,即函数渐上界,反映当 <span class="arithmatex">\(n\)</span>正无穷时,<span class="arithmatex">\(T(n)\)</span> 增长级别。</li>
<li>推算时间复杂度分为两步,首先统计计算操作数量,然后判断渐上界。</li>
<li>常见时间复杂度从小到大排列有 <span class="arithmatex">\(O(1)\)</span> , <span class="arithmatex">\(O(\log n)\)</span> , <span class="arithmatex">\(O(n)\)</span> , <span class="arithmatex">\(O(n \log n)\)</span> , <span class="arithmatex">\(O(n^2)\)</span> , <span class="arithmatex">\(O(2^n)\)</span> , <span class="arithmatex">\(O(n!)\)</span> </li>
<li>某些算法的时间复杂度非固定,而是与输入数据的分布有关。时间复杂度分为最差、最佳、平均时间复杂度最佳时间复杂度几乎不用,因为输入数据一般需要满足严格条件才能达到最佳情况。</li>
<li>平均时间复杂度反映算法在随机数据输入下的运行效率,最接近实际应用中的算法性能。计算平均时间复杂度需要统计输入数据分布以及综合后的数学期望。</li>
</ul>
<h3 id="_3">空间复杂度<a class="headerlink" href="#_3" title="Permanent link">&para;</a></h3>
<ul>
<li>时间复杂度的定义类似,空间复杂度统计算法占用空间随数据量变大时的增长趋势。</li>
<li>算法运行相关内存空间可分为输入空间、暂存空间、输出空间。通常情况下,输入空间不计入空间复杂度计算。暂存空间可分为指令空间、数据空间、栈帧空间,其中栈帧空间一般在递归函数中才会影响空间复杂度。</li>
<li>我们一般只关最差空间复杂度,即统计算法在最差输入数据和最差运行时间点下的空间复杂度。</li>
<li>常见空间复杂度从小到大排列有 <span class="arithmatex">\(O(1)\)</span> , <span class="arithmatex">\(O(\log n)\)</span> , <span class="arithmatex">\(O(n)\)</span> , <span class="arithmatex">\(O(n^2)\)</span> , <span class="arithmatex">\(O(2^n)\)</span></li>
<li>类似于时间复杂度,空间复杂度用于衡量算法占用空间随数据量增长趋势。</li>
<li>算法运行过程中的相关内存空间可分为输入空间、暂存空间、输出空间。通常情况下,输入空间不计入空间复杂度计算。暂存空间可分为指令空间、数据空间、栈帧空间,其中栈帧空间通常仅在递归函数中影响空间复杂度。</li>
<li>我们通常只关最差空间复杂度,即统计算法在最差输入数据和最差运行时间点下的空间复杂度。</li>
<li>常见空间复杂度从小到大排列有 <span class="arithmatex">\(O(1)\)</span> , <span class="arithmatex">\(O(\log n)\)</span> , <span class="arithmatex">\(O(n)\)</span> , <span class="arithmatex">\(O(n^2)\)</span> , <span class="arithmatex">\(O(2^n)\)</span> </li>
</ul>

View File

@ -1947,11 +1947,11 @@
<h1 id="22">2.2. &nbsp; 时间复杂度<a class="headerlink" href="#22" title="Permanent link">&para;</a></h1>
<h2 id="221">2.2.1. &nbsp; 统计算法运行时间<a class="headerlink" href="#221" title="Permanent link">&para;</a></h2>
<p>运行时间能够直观且准确地体现出算法的效率水平。如果我们想要 <strong>准确预估一段代码的运行时间</strong> 该如何呢?</p>
<p>运行时间可以直观且准确地反映算法的效率。然而,如果我们想要准确预估一段代码的运行时间,应该如何操作呢?</p>
<ol>
<li>首先需要 <strong>确定运行平台</strong> ,包括硬件配置、编程语言、系统环境等,这些都会影响代码的运行效率。</li>
<li>评估 <strong>各种计算操作所需运行时间</strong> ,例如加法操作 <code>+</code> 需要 1 ns ,乘法操作 <code>*</code> 需要 10 ns ,打印操作需要 5 ns 等。</li>
<li>根据代码 <strong>统计所有计算操作的数量</strong> ,并将所有操作的执行时间求和,即可得到运行时间。</li>
<li><strong>确定运行平台</strong>,包括硬件配置、编程语言、系统环境等,这些因素都会影响代码的运行效率。</li>
<li><strong>评估各种计算操作所需运行时间</strong>,例如加法操作 <code>+</code> 需要 1 ns乘法操作 <code>*</code> 需要 10 ns打印操作需要 5 ns 等。</li>
<li><strong>统计代码中所有计算操作</strong>,并将所有操作的执行时间求和,从而得到运行时间。</li>
</ol>
<p>例如以下代码,输入数据大小为 <span class="arithmatex">\(n\)</span> ,根据以上方法,可以得到算法运行时间为 <span class="arithmatex">\(6n + 12\)</span> ns 。</p>
<div class="arithmatex">\[
@ -2082,13 +2082,13 @@
</div>
</div>
</div>
<p>实际上, <strong>统计算法的运行时间既不合理也不现实</strong>。首先,我们不希望预估时间和运行平台绑定,毕竟算法需要在各式各样的平台之上。其次,我们很难获知每种操作的运行时间,这预估过程带来了极大的难度。</p>
<p>然而实际上,<strong>统计算法的运行时间既不合理也不现实</strong>。首先,我们不希望预估时间和运行平台绑定,因为算法需要在各种不同的平台上运行。其次,我们很难获知每种操作的运行时间,这预估过程带来了极大的难度。</p>
<h2 id="222">2.2.2. &nbsp; 统计时间增长趋势<a class="headerlink" href="#222" title="Permanent link">&para;</a></h2>
<p>「时间复杂度分析」采取了不同的法,其统计的不是算法运行时间,而是 <strong>算法运行时间随着数据量变大时的增长趋势</strong> </p>
<p>“时间增长趋势”这个概念较抽象,我们借助一个例子来理解。设输入数据大小为 <span class="arithmatex">\(n\)</span> ,给定三个算法 <code>A</code> , <code>B</code> , <code>C</code></p>
<p>「时间复杂度分析」采取了一种不同的法,其统计的不是算法运行时间,<strong>而是算法运行时间随着数据量变大时的增长趋势</strong></p>
<p>“时间增长趋势”这个概念较抽象,我们通过一个例子来加以理解。设输入数据大小为 <span class="arithmatex">\(n\)</span>,给定三个算法 <code>A</code><code>B</code><code>C</code></p>
<ul>
<li>算法 <code>A</code> 只有 <span class="arithmatex">\(1\)</span> 个打印操作,算法运行时间不随着 <span class="arithmatex">\(n\)</span> 增大而增长。我们称此算法的时间复杂度为「常数阶」。</li>
<li>算法 <code>B</code> 中的打印操作需要循环 <span class="arithmatex">\(n\)</span> 次,算法运行时间随着 <span class="arithmatex">\(n\)</span> 增大线性增长。此算法的时间复杂度被称为「线性阶」。</li>
<li>算法 <code>B</code> 中的打印操作需要循环 <span class="arithmatex">\(n\)</span> 次,算法运行时间随着 <span class="arithmatex">\(n\)</span> 增大线性增长。此算法的时间复杂度被称为「线性阶」。</li>
<li>算法 <code>C</code> 中的打印操作需要循环 <span class="arithmatex">\(1000000\)</span> 次,但运行时间仍与输入数据大小 <span class="arithmatex">\(n\)</span> 无关。因此 <code>C</code> 的时间复杂度和 <code>A</code> 相同,仍为「常数阶」。</li>
</ul>
<div class="tabbed-set tabbed-alternate" data-tabs="2:10"><input checked="checked" id="__tabbed_2_1" name="__tabbed_2" type="radio" /><input id="__tabbed_2_2" name="__tabbed_2" type="radio" /><input id="__tabbed_2_3" name="__tabbed_2" type="radio" /><input id="__tabbed_2_4" name="__tabbed_2" type="radio" /><input id="__tabbed_2_5" name="__tabbed_2" type="radio" /><input id="__tabbed_2_6" name="__tabbed_2" type="radio" /><input id="__tabbed_2_7" name="__tabbed_2" type="radio" /><input id="__tabbed_2_8" name="__tabbed_2" type="radio" /><input id="__tabbed_2_9" name="__tabbed_2" type="radio" /><input id="__tabbed_2_10" name="__tabbed_2" type="radio" /><div class="tabbed-labels"><label for="__tabbed_2_1">Java</label><label for="__tabbed_2_2">C++</label><label for="__tabbed_2_3">Python</label><label for="__tabbed_2_4">Go</label><label for="__tabbed_2_5">JavaScript</label><label for="__tabbed_2_6">TypeScript</label><label for="__tabbed_2_7">C</label><label for="__tabbed_2_8">C#</label><label for="__tabbed_2_9">Swift</label><label for="__tabbed_2_10">Zig</label></div>
@ -2275,12 +2275,12 @@
<p><img alt="算法 A, B, C 的时间增长趋势" src="../time_complexity.assets/time_complexity_simple_example.png" /></p>
<p align="center"> Fig. 算法 A, B, C 的时间增长趋势 </p>
<p>直接统计算法运行时间,时间复杂度分析的做法有什么好处呢?以及有什么不足</p>
<p><strong>时间复杂度可以有效评估算法效率</strong>。算法 <code>B</code> 运行时间的增长是线性的,在 <span class="arithmatex">\(n &gt; 1\)</span>慢于算法 <code>A</code> ,在 <span class="arithmatex">\(n &gt; 1000000\)</span>慢于算法 <code>C</code> 。实质上,只要输入数据大小 <span class="arithmatex">\(n\)</span> 足够大,复杂度为「常数阶」的算法一定优于「线性阶」的算法,这正是时间增长趋势的含义。</p>
<p><strong>时间复杂度的推算方法更简便</strong>在时间复杂度分析中,我们可以将统计「计算操作的运行时间」简化为统计「计算操作的数量」,这是因为,无论是运行平台还是计算操作类型都与算法运行时间的增长趋势无关。因,我们可以简单地将所有计算操作的执行时间统一看作是相同的“单位时间”,这样的简化法大大降低了估算难度。</p>
<p><strong>时间复杂度也存在一定的局限性</strong>如,虽然算法 <code>A</code><code>C</code> 的时间复杂度相同,但实际运行时间有非常大的差别。再比如,虽然算法 <code>B</code><code>C</code> 的时间复杂度要更高,但在输入数据大小 <span class="arithmatex">\(n\)</span> 较小时,算法 <code>B</code> 是要明显优于算法 <code>C</code> 的。对于以上情况,我们很难仅凭时间复杂度来判定算法效率高低。然而,即使存在这些问题,复杂度分析仍然是评判算法效率最有效且常用的方法。</p>
<p>较于直接统计算法运行时间,时间复杂度分析有哪些优势和局限性呢</p>
<p><strong>时间复杂度能够有效评估算法效率</strong>例如,算法 <code>B</code> 运行时间呈线性增长,在 <span class="arithmatex">\(n &gt; 1\)</span>算法 <code>A</code> ,在 <span class="arithmatex">\(n &gt; 1000000\)</span>算法 <code>C</code> 慢。事实上,只要输入数据大小 <span class="arithmatex">\(n\)</span> 足够大,复杂度为「常数阶」的算法一定优于「线性阶」的算法,这正是时间增长趋势所表达的含义。</p>
<p><strong>时间复杂度的推算方法更简便</strong>显然,运行平台计算操作类型都与算法运行时间的增长趋势无关。因此在时间复杂度分析中,我们可以简单地将所有计算操作的执行时间视为相同的“单位时间”,从而将“计算操作的运行时间的统计”简化为“计算操作的数量的统计”,这样的简化法大大降低了估算难度。</p>
<p><strong>时间复杂度也存在一定的局限性</strong>如,尽管算法 <code>A</code><code>C</code> 的时间复杂度相同,但实际运行时间差别很大。同样,尽管算法 <code>B</code> 的时间复杂度<code>C</code> 高,但在输入数据大小 <span class="arithmatex">\(n\)</span> 较小时,算法 <code>B</code> 明显优于算法 <code>C</code> 。在这些情况,我们很难仅凭时间复杂度判断算法效率高低。当然,尽管存在上述问题,复杂度分析仍然是评判算法效率最有效且常用的方法。</p>
<h2 id="223">2.2.3. &nbsp; 函数渐近上界<a class="headerlink" href="#223" title="Permanent link">&para;</a></h2>
<p>设算法计算操作数量」为 <span class="arithmatex">\(T(n)\)</span> ,其是一个关于输入数据大小 <span class="arithmatex">\(n\)</span> 的函数。例如,以下算法的操作数量为</p>
<p>设算法计算操作数量是一个关于输入数据大小 <span class="arithmatex">\(n\)</span> 的函数,记为 <span class="arithmatex">\(T(n)\)</span> 以下算法的操作数量为</p>
<div class="arithmatex">\[
T(n) = 3 + 2n
\]</div>
@ -2400,9 +2400,9 @@ T(n) = 3 + 2n
</div>
</div>
</div>
<p><span class="arithmatex">\(T(n)\)</span>一次函数,说明时间增长趋势是线性的,因此易得时间复杂度是线性阶。</p>
<p>我们将线性阶的时间复杂度记为 <span class="arithmatex">\(O(n)\)</span> ,这个数学符号称为「大 <span class="arithmatex">\(O\)</span> 记号 Big-<span class="arithmatex">\(O\)</span> Notation」表函数 <span class="arithmatex">\(T(n)\)</span> 的「渐近上界 asymptotic upper bound」。</p>
<p>我们要推算时间复杂度本质上是计算操作数量函数 <span class="arithmatex">\(T(n)\)</span>的渐近上界。下面我们先来看看函数渐近上界的数学定义。</p>
<p><span class="arithmatex">\(T(n)\)</span> 是一次函数,说明时间增长趋势是线性的,因此可以得出时间复杂度是线性阶。</p>
<p>我们将线性阶的时间复杂度记为 <span class="arithmatex">\(O(n)\)</span> ,这个数学符号称为「大 <span class="arithmatex">\(O\)</span> 记号 Big-<span class="arithmatex">\(O\)</span> Notation」函数 <span class="arithmatex">\(T(n)\)</span> 的「渐近上界 Asymptotic Upper Bound」。</p>
<p>推算时间复杂度本质上是计算操作数量函数 <span class="arithmatex">\(T(n)\)</span>的渐近上界。接下来,我们来看函数渐近上界的数学定义。</p>
<div class="admonition abstract">
<p class="admonition-title">函数渐近上界</p>
<p>若存在正实数 <span class="arithmatex">\(c\)</span> 和实数 <span class="arithmatex">\(n_0\)</span> ,使得对于所有的 <span class="arithmatex">\(n &gt; n_0\)</span> ,均有
@ -2417,18 +2417,15 @@ $$</p>
<p><img alt="函数的渐近上界" src="../time_complexity.assets/asymptotic_upper_bound.png" /></p>
<p align="center"> Fig. 函数的渐近上界 </p>
<p>本质上,计算渐近上界就是找一个函数 <span class="arithmatex">\(f(n)\)</span> <strong>使得 <span class="arithmatex">\(n\)</span> 趋向于无穷大时,<span class="arithmatex">\(T(n)\)</span><span class="arithmatex">\(f(n)\)</span> 处于相同的增长级别仅相差一个常数项 <span class="arithmatex">\(c\)</span> 的倍数</strong></p>
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p>渐近上界的数学味儿有点重,如果你感觉没有完全理解,无需担心,因为在实际使用中我们只需要会推算即可,数学意义可以慢慢领悟。</p>
</div>
<p>本质上,计算渐近上界就是找一个函数 <span class="arithmatex">\(f(n)\)</span> ,使得 <span class="arithmatex">\(n\)</span> 趋向于无穷大时,<span class="arithmatex">\(T(n)\)</span><span class="arithmatex">\(f(n)\)</span> 处于相同的增长级别仅相差一个常数项 <span class="arithmatex">\(c\)</span> 的倍数。</p>
<h2 id="224">2.2.4. &nbsp; 推算方法<a class="headerlink" href="#224" title="Permanent link">&para;</a></h2>
<p>推算出 <span class="arithmatex">\(f(n)\)</span> 后,我们就得到时间复杂度 <span class="arithmatex">\(O(f(n))\)</span> 。那么,如何来确定渐近上界 <span class="arithmatex">\(f(n)\)</span> 呢?总体分为两步,首先「统计操作数量」,然后「判断渐近上界」</p>
<p>渐近上界的数学味儿有点重,如果你感觉没有完全理解,也无需担心。因为在实际使用中,我们只需要掌握推算方法,数学意义可以逐渐领悟</p>
<p>根据定义,确定 <span class="arithmatex">\(f(n)\)</span> 之后,我们便可得到时间复杂度 <span class="arithmatex">\(O(f(n))\)</span> 。那么如何确定渐近上界 <span class="arithmatex">\(f(n)\)</span> 呢?总体分为两步:首先统计操作数量,然后判断渐近上界。</p>
<h3 id="1">1) 统计操作数量<a class="headerlink" href="#1" title="Permanent link">&para;</a></h3>
<p>代码,从上到下一行一行地计数即可。然而,<strong>由于上述 <span class="arithmatex">\(c \cdot f(n)\)</span> 中的常数项 <span class="arithmatex">\(c\)</span> 可以取任意大小,因此操作数量 <span class="arithmatex">\(T(n)\)</span> 中的各种系数、常数项都可以被忽略</strong>。根据此原则,可以总结出以下计数偷懒技巧:</p>
<p>对代码,逐行从上到下计算即可。然而,由于上述 <span class="arithmatex">\(c \cdot f(n)\)</span> 中的常数项 <span class="arithmatex">\(c\)</span> 可以取任意大小,<strong>因此操作数量 <span class="arithmatex">\(T(n)\)</span> 中的各种系数、常数项都可以被忽略</strong>。根据此原则,可以总结出以下计数简化技巧:</p>
<ol>
<li><strong>跳过数量<span class="arithmatex">\(n\)</span> 无关的操作</strong>。因为们都是 <span class="arithmatex">\(T(n)\)</span> 中的常数项,对时间复杂度不产生影响。</li>
<li><strong>省略所有系数</strong>。例如,循环 <span class="arithmatex">\(2n\)</span> 次、<span class="arithmatex">\(5n + 1\)</span>、……,都可以简记为 <span class="arithmatex">\(n\)</span> 次,因为 <span class="arithmatex">\(n\)</span> 前面的系数对时间复杂度也不产生影响。</li>
<li><strong>忽略<span class="arithmatex">\(n\)</span> 无关的操作</strong>。因为们都是 <span class="arithmatex">\(T(n)\)</span> 中的常数项,对时间复杂度不产生影响。</li>
<li><strong>省略所有系数</strong>。例如,循环 <span class="arithmatex">\(2n\)</span> 次、<span class="arithmatex">\(5n + 1\)</span>,都可以简记为 <span class="arithmatex">\(n\)</span> 次,因为 <span class="arithmatex">\(n\)</span> 前面的系数对时间复杂度没有影响。</li>
<li><strong>循环嵌套时使用乘法</strong>。总操作数量等于外层循环和内层循环操作数量之积,每一层循环依然可以分别套用上述 <code>1.</code><code>2.</code> 技巧。</li>
</ol>
<p>以下示例展示了使用上述技巧前、后的统计结果。</p>
@ -2602,8 +2599,8 @@ T(n) &amp; = n^2 + n &amp; \text{偷懒统计 (o.O)}
</div>
</div>
<h3 id="2">2) 判断渐近上界<a class="headerlink" href="#2" title="Permanent link">&para;</a></h3>
<p><strong>时间复杂度由多项式 <span class="arithmatex">\(T(n)\)</span> 中最高阶的项来决定</strong>。这是因为在 <span class="arithmatex">\(n\)</span> 趋于无穷大时,最高阶的项将处于主导作用,其它项的影响都可以被忽略。</p>
<p>以下表格给出了一些例子,其中一些夸张的值,是想要向大家强调 <strong>系数无法撼动阶数</strong> 这一结论。 <span class="arithmatex">\(n\)</span> 趋于无穷大时,这些常数都是“浮云”</p>
<p><strong>时间复杂度由多项式 <span class="arithmatex">\(T(n)\)</span> 中最高阶的项来决定</strong>。这是因为在 <span class="arithmatex">\(n\)</span> 趋于无穷大时,最高阶的项将发挥主导作用,其它项的影响都可以被忽略。</p>
<p>以下表格展示了一些例子,其中一些夸张的值是为了强调“系数无法撼动阶数”这一结论。 <span class="arithmatex">\(n\)</span> 趋于无穷大时,这些常数变得无足轻重</p>
<div class="center-table">
<table>
<thead>
@ -2637,7 +2634,7 @@ T(n) &amp; = n^2 + n &amp; \text{偷懒统计 (o.O)}
</table>
</div>
<h2 id="225">2.2.5. &nbsp; 常见类型<a class="headerlink" href="#225" title="Permanent link">&para;</a></h2>
<p>设输入数据大小为 <span class="arithmatex">\(n\)</span> ,常见的时间复杂度类型有(从低到高排列)</p>
<p>设输入数据大小为 <span class="arithmatex">\(n\)</span> ,常见的时间复杂度类型包括(按照从低到高的顺序排列)</p>
<div class="arithmatex">\[
\begin{aligned}
O(1) &lt; O(\log n) &lt; O(n) &lt; O(n \log n) &lt; O(n^2) &lt; O(2^n) &lt; O(n!) \newline
@ -2649,11 +2646,11 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n \log n) &lt; O(n^2) &lt; O(2^n) &lt; O(n!
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p>部分示例代码需要一些前置知识,包括数组、递归算法等。如果遇到看不懂的地方无需担心,可以在学习完后面章节后再来复习,现阶段先聚焦在理解时间复杂度含义和推算方法</p>
<p>部分示例代码需要一些预备知识,包括数组、递归算法等。如果遇到不理解的部分,请不要担心,可以在学习完后面章节后再回顾。现阶段,请先专注于理解时间复杂度含义和推算方法。</p>
</div>
<h3 id="o1">常数阶 <span class="arithmatex">\(O(1)\)</span><a class="headerlink" href="#o1" title="Permanent link">&para;</a></h3>
<p>常数阶的操作数量与输入数据大小 <span class="arithmatex">\(n\)</span> 无关,即不随着 <span class="arithmatex">\(n\)</span> 的变化而变化。</p>
<p>对于以下算法,无论操作数量 <code>size</code> 有多大,只要与数据大小 <span class="arithmatex">\(n\)</span> 无关,时间复杂度仍为 <span class="arithmatex">\(O(1)\)</span></p>
<p>对于以下算法,尽管操作数量 <code>size</code> 可能很大,但由于其与数据大小 <span class="arithmatex">\(n\)</span> 无关,因此时间复杂度仍为 <span class="arithmatex">\(O(1)\)</span></p>
<div class="tabbed-set tabbed-alternate" data-tabs="5:10"><input checked="checked" id="__tabbed_5_1" name="__tabbed_5" type="radio" /><input id="__tabbed_5_2" name="__tabbed_5" type="radio" /><input id="__tabbed_5_3" name="__tabbed_5" type="radio" /><input id="__tabbed_5_4" name="__tabbed_5" type="radio" /><input id="__tabbed_5_5" name="__tabbed_5" type="radio" /><input id="__tabbed_5_6" name="__tabbed_5" type="radio" /><input id="__tabbed_5_7" name="__tabbed_5" type="radio" /><input id="__tabbed_5_8" name="__tabbed_5" type="radio" /><input id="__tabbed_5_9" name="__tabbed_5" type="radio" /><input id="__tabbed_5_10" name="__tabbed_5" type="radio" /><div class="tabbed-labels"><label for="__tabbed_5_1">Java</label><label for="__tabbed_5_2">C++</label><label for="__tabbed_5_3">Python</label><label for="__tabbed_5_4">Go</label><label for="__tabbed_5_5">JavaScript</label><label for="__tabbed_5_6">TypeScript</label><label for="__tabbed_5_7">C</label><label for="__tabbed_5_8">C#</label><label for="__tabbed_5_9">Swift</label><label for="__tabbed_5_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2765,7 +2762,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n \log n) &lt; O(n^2) &lt; O(2^n) &lt; O(n!
</div>
</div>
<h3 id="on">线性阶 <span class="arithmatex">\(O(n)\)</span><a class="headerlink" href="#on" title="Permanent link">&para;</a></h3>
<p>线性阶的操作数量相对输入数据大小线性级别增长。线性阶常出现单层循环。</p>
<p>线性阶的操作数量相对输入数据大小线性级别增长。线性阶常出现单层循环</p>
<div class="tabbed-set tabbed-alternate" data-tabs="6:10"><input checked="checked" id="__tabbed_6_1" name="__tabbed_6" type="radio" /><input id="__tabbed_6_2" name="__tabbed_6" type="radio" /><input id="__tabbed_6_3" name="__tabbed_6" type="radio" /><input id="__tabbed_6_4" name="__tabbed_6" type="radio" /><input id="__tabbed_6_5" name="__tabbed_6" type="radio" /><input id="__tabbed_6_6" name="__tabbed_6" type="radio" /><input id="__tabbed_6_7" name="__tabbed_6" type="radio" /><input id="__tabbed_6_8" name="__tabbed_6" type="radio" /><input id="__tabbed_6_9" name="__tabbed_6" type="radio" /><input id="__tabbed_6_10" name="__tabbed_6" type="radio" /><div class="tabbed-labels"><label for="__tabbed_6_1">Java</label><label for="__tabbed_6_2">C++</label><label for="__tabbed_6_3">Python</label><label for="__tabbed_6_4">Go</label><label for="__tabbed_6_5">JavaScript</label><label for="__tabbed_6_6">TypeScript</label><label for="__tabbed_6_7">C</label><label for="__tabbed_6_8">C#</label><label for="__tabbed_6_9">Swift</label><label for="__tabbed_6_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -2866,10 +2863,10 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n \log n) &lt; O(n^2) &lt; O(2^n) &lt; O(n!
</div>
</div>
</div>
<p>遍历数组」和「遍历链表等操作时间复杂度<span class="arithmatex">\(O(n)\)</span> ,其中 <span class="arithmatex">\(n\)</span> 为数组或链表的长度。</p>
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p><strong>数据大小 <span class="arithmatex">\(n\)</span> 根据输入数据的类型来确定</strong>如,在上述示例中,我们直接将 <span class="arithmatex">\(n\)</span> 看作输入数据大小;以下遍历数组示例中,数据大小 <span class="arithmatex">\(n\)</span> 为数组的长度。</p>
<p>遍历数组遍历链表等操作时间复杂度<span class="arithmatex">\(O(n)\)</span> ,其中 <span class="arithmatex">\(n\)</span> 为数组或链表的长度。</p>
<div class="admonition question">
<p class="admonition-title">如何确定输入数据大小 <span class="arithmatex">\(n\)</span> </p>
<p><strong>数据大小 <span class="arithmatex">\(n\)</span> 根据输入数据的类型来具体确定</strong>如,在上述示例中,我们直接将 <span class="arithmatex">\(n\)</span> 视为输入数据大小;在下面遍历数组示例中,数据大小 <span class="arithmatex">\(n\)</span> 为数组的长度。</p>
</div>
<div class="tabbed-set tabbed-alternate" data-tabs="7:10"><input checked="checked" id="__tabbed_7_1" name="__tabbed_7" type="radio" /><input id="__tabbed_7_2" name="__tabbed_7" type="radio" /><input id="__tabbed_7_3" name="__tabbed_7" type="radio" /><input id="__tabbed_7_4" name="__tabbed_7" type="radio" /><input id="__tabbed_7_5" name="__tabbed_7" type="radio" /><input id="__tabbed_7_6" name="__tabbed_7" type="radio" /><input id="__tabbed_7_7" name="__tabbed_7" type="radio" /><input id="__tabbed_7_8" name="__tabbed_7" type="radio" /><input id="__tabbed_7_9" name="__tabbed_7" type="radio" /><input id="__tabbed_7_10" name="__tabbed_7" type="radio" /><div class="tabbed-labels"><label for="__tabbed_7_1">Java</label><label for="__tabbed_7_2">C++</label><label for="__tabbed_7_3">Python</label><label for="__tabbed_7_4">Go</label><label for="__tabbed_7_5">JavaScript</label><label for="__tabbed_7_6">TypeScript</label><label for="__tabbed_7_7">C</label><label for="__tabbed_7_8">C#</label><label for="__tabbed_7_9">Swift</label><label for="__tabbed_7_10">Zig</label></div>
<div class="tabbed-content">
@ -2988,7 +2985,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n \log n) &lt; O(n^2) &lt; O(2^n) &lt; O(n!
</div>
</div>
<h3 id="on2">平方阶 <span class="arithmatex">\(O(n^2)\)</span><a class="headerlink" href="#on2" title="Permanent link">&para;</a></h3>
<p>平方阶的操作数量相对输入数据大小平方级别增长。平方阶常出现嵌套循环,外层循环和内层循环都为 <span class="arithmatex">\(O(n)\)</span> ,总体为 <span class="arithmatex">\(O(n^2)\)</span></p>
<p>平方阶的操作数量相对输入数据大小平方级别增长。平方阶常出现嵌套循环,外层循环和内层循环都为 <span class="arithmatex">\(O(n)\)</span> 因此总体为 <span class="arithmatex">\(O(n^2)\)</span></p>
<div class="tabbed-set tabbed-alternate" data-tabs="8:10"><input checked="checked" id="__tabbed_8_1" name="__tabbed_8" type="radio" /><input id="__tabbed_8_2" name="__tabbed_8" type="radio" /><input id="__tabbed_8_3" name="__tabbed_8" type="radio" /><input id="__tabbed_8_4" name="__tabbed_8" type="radio" /><input id="__tabbed_8_5" name="__tabbed_8" type="radio" /><input id="__tabbed_8_6" name="__tabbed_8" type="radio" /><input id="__tabbed_8_7" name="__tabbed_8" type="radio" /><input id="__tabbed_8_8" name="__tabbed_8" type="radio" /><input id="__tabbed_8_9" name="__tabbed_8" type="radio" /><input id="__tabbed_8_10" name="__tabbed_8" type="radio" /><div class="tabbed-labels"><label for="__tabbed_8_1">Java</label><label for="__tabbed_8_2">C++</label><label for="__tabbed_8_3">Python</label><label for="__tabbed_8_4">Go</label><label for="__tabbed_8_5">JavaScript</label><label for="__tabbed_8_6">TypeScript</label><label for="__tabbed_8_7">C</label><label for="__tabbed_8_8">C#</label><label for="__tabbed_8_9">Swift</label><label for="__tabbed_8_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -3128,7 +3125,7 @@ O(1) &lt; O(\log n) &lt; O(n) &lt; O(n \log n) &lt; O(n^2) &lt; O(2^n) &lt; O(n!
<p><img alt="常数阶、线性阶、平方阶的时间复杂度" src="../time_complexity.assets/time_complexity_constant_linear_quadratic.png" /></p>
<p align="center"> Fig. 常数阶、线性阶、平方阶的时间复杂度 </p>
<p>以「冒泡排序」为例,外层循环 <span class="arithmatex">\(n - 1\)</span> 次,内层循环 <span class="arithmatex">\(n-1, n-2, \cdots, 2, 1\)</span> 次,平均为 <span class="arithmatex">\(\frac{n}{2}\)</span> 次,因此时间复杂度为 <span class="arithmatex">\(O(n^2)\)</span></p>
<p>以「冒泡排序」为例,外层循环执行 <span class="arithmatex">\(n - 1\)</span> 次,内层循环执行 <span class="arithmatex">\(n-1, n-2, \cdots, 2, 1\)</span> 次,平均为 <span class="arithmatex">\(\frac{n}{2}\)</span> 次,因此时间复杂度为 <span class="arithmatex">\(O(n^2)\)</span></p>
<div class="arithmatex">\[
O((n - 1) \frac{n}{2}) = O(n^2)
\]</div>
@ -3334,9 +3331,9 @@ O((n - 1) \frac{n}{2}) = O(n^2)
<h3 id="o2n">指数阶 <span class="arithmatex">\(O(2^n)\)</span><a class="headerlink" href="#o2n" title="Permanent link">&para;</a></h3>
<div class="admonition note">
<p class="admonition-title">Note</p>
<p>生物学科中的“细胞分裂”是指数阶增长:初始状态为 <span class="arithmatex">\(1\)</span> 个细胞,分裂一轮后为 <span class="arithmatex">\(2\)</span> 个,分裂两轮后为 <span class="arithmatex">\(4\)</span> 个,……,分裂 <span class="arithmatex">\(n\)</span> 轮后有 <span class="arithmatex">\(2^n\)</span> 个细胞。</p>
<p>生物学的“细胞分裂”是指数阶增长的典型例子:初始状态为 <span class="arithmatex">\(1\)</span> 个细胞,分裂一轮后<span class="arithmatex">\(2\)</span> 个,分裂两轮后<span class="arithmatex">\(4\)</span> 个,以此类推,分裂 <span class="arithmatex">\(n\)</span> 轮后有 <span class="arithmatex">\(2^n\)</span> 个细胞。</p>
</div>
<p>指数阶增长非常,在实际应用中一般是不能被接受的。若一个问题使用「暴力枚举」求解的时间复杂度 <span class="arithmatex">\(O(2^n)\)</span> ,那么一般都需要使用「动态规划」或「贪心算法」等法来解。</p>
<p>指数阶增长非常迅速,在实际应用中通常是不可接受的。若一个问题使用「暴力枚举」求解的时间复杂度 <span class="arithmatex">\(O(2^n)\)</span> ,那么通常需要使用「动态规划」或「贪心算法」等法来解</p>
<div class="tabbed-set tabbed-alternate" data-tabs="10:10"><input checked="checked" id="__tabbed_10_1" name="__tabbed_10" type="radio" /><input id="__tabbed_10_2" name="__tabbed_10" type="radio" /><input id="__tabbed_10_3" name="__tabbed_10" type="radio" /><input id="__tabbed_10_4" name="__tabbed_10" type="radio" /><input id="__tabbed_10_5" name="__tabbed_10" type="radio" /><input id="__tabbed_10_6" name="__tabbed_10" type="radio" /><input id="__tabbed_10_7" name="__tabbed_10" type="radio" /><input id="__tabbed_10_8" name="__tabbed_10" type="radio" /><input id="__tabbed_10_9" name="__tabbed_10" type="radio" /><input id="__tabbed_10_10" name="__tabbed_10" type="radio" /><div class="tabbed-labels"><label for="__tabbed_10_1">Java</label><label for="__tabbed_10_2">C++</label><label for="__tabbed_10_3">Python</label><label for="__tabbed_10_4">Go</label><label for="__tabbed_10_5">JavaScript</label><label for="__tabbed_10_6">TypeScript</label><label for="__tabbed_10_7">C</label><label for="__tabbed_10_8">C#</label><label for="__tabbed_10_9">Swift</label><label for="__tabbed_10_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -3499,7 +3496,7 @@ O((n - 1) \frac{n}{2}) = O(n^2)
<p><img alt="指数阶的时间复杂度" src="../time_complexity.assets/time_complexity_exponential.png" /></p>
<p align="center"> Fig. 指数阶的时间复杂度 </p>
<p>在实际算法中,指数阶常出现于递归函数。例如以下代码,不断地一分为二,分裂 <span class="arithmatex">\(n\)</span> 次后停止。</p>
<p>在实际算法中,指数阶常出现于递归函数。例如以下代码,不断地一分为二,经过 <span class="arithmatex">\(n\)</span>分裂后停止。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="11:10"><input checked="checked" id="__tabbed_11_1" name="__tabbed_11" type="radio" /><input id="__tabbed_11_2" name="__tabbed_11" type="radio" /><input id="__tabbed_11_3" name="__tabbed_11" type="radio" /><input id="__tabbed_11_4" name="__tabbed_11" type="radio" /><input id="__tabbed_11_5" name="__tabbed_11" type="radio" /><input id="__tabbed_11_6" name="__tabbed_11" type="radio" /><input id="__tabbed_11_7" name="__tabbed_11" type="radio" /><input id="__tabbed_11_8" name="__tabbed_11" type="radio" /><input id="__tabbed_11_9" name="__tabbed_11" type="radio" /><input id="__tabbed_11_10" name="__tabbed_11" type="radio" /><div class="tabbed-labels"><label for="__tabbed_11_1">Java</label><label for="__tabbed_11_2">C++</label><label for="__tabbed_11_3">Python</label><label for="__tabbed_11_4">Go</label><label for="__tabbed_11_5">JavaScript</label><label for="__tabbed_11_6">TypeScript</label><label for="__tabbed_11_7">C</label><label for="__tabbed_11_8">C#</label><label for="__tabbed_11_9">Swift</label><label for="__tabbed_11_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -3585,8 +3582,8 @@ O((n - 1) \frac{n}{2}) = O(n^2)
</div>
</div>
<h3 id="olog-n">对数阶 <span class="arithmatex">\(O(\log n)\)</span><a class="headerlink" href="#olog-n" title="Permanent link">&para;</a></h3>
<p>对数阶与指数阶正好相反,后者反映“每轮增加到两倍的情况”,而前者反映“每轮缩减到一半的情况”。对数阶仅次于常数阶,时间增长得很慢,是理想的时间复杂度。</p>
<p>对数阶常出现于「二分查找」和「分治算法」中,体现“一分为多”“化繁为简”的算法思想。</p>
<p>与指数阶相反,对数阶反映“每轮缩减到一半的情况”。对数阶仅次于常数阶,时间增长慢,是理想的时间复杂度。</p>
<p>对数阶常出现于「二分查找」和「分治算法」中,体现“一分为多”“化繁为简”的算法思想。</p>
<p>设输入数据大小为 <span class="arithmatex">\(n\)</span> ,由于每轮缩减到一半,因此循环次数是 <span class="arithmatex">\(\log_2 n\)</span> ,即 <span class="arithmatex">\(2^n\)</span> 的反函数。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="12:10"><input checked="checked" id="__tabbed_12_1" name="__tabbed_12" type="radio" /><input id="__tabbed_12_2" name="__tabbed_12" type="radio" /><input id="__tabbed_12_3" name="__tabbed_12" type="radio" /><input id="__tabbed_12_4" name="__tabbed_12" type="radio" /><input id="__tabbed_12_5" name="__tabbed_12" type="radio" /><input id="__tabbed_12_6" name="__tabbed_12" type="radio" /><input id="__tabbed_12_7" name="__tabbed_12" type="radio" /><input id="__tabbed_12_8" name="__tabbed_12" type="radio" /><input id="__tabbed_12_9" name="__tabbed_12" type="radio" /><input id="__tabbed_12_10" name="__tabbed_12" type="radio" /><div class="tabbed-labels"><label for="__tabbed_12_1">Java</label><label for="__tabbed_12_2">C++</label><label for="__tabbed_12_3">Python</label><label for="__tabbed_12_4">Go</label><label for="__tabbed_12_5">JavaScript</label><label for="__tabbed_12_6">TypeScript</label><label for="__tabbed_12_7">C</label><label for="__tabbed_12_8">C#</label><label for="__tabbed_12_9">Swift</label><label for="__tabbed_12_10">Zig</label></div>
<div class="tabbed-content">
@ -3797,7 +3794,7 @@ O((n - 1) \frac{n}{2}) = O(n^2)
</div>
<h3 id="on-log-n">线性对数阶 <span class="arithmatex">\(O(n \log n)\)</span><a class="headerlink" href="#on-log-n" title="Permanent link">&para;</a></h3>
<p>线性对数阶常出现于嵌套循环中,两层循环的时间复杂度分别为 <span class="arithmatex">\(O(\log n)\)</span><span class="arithmatex">\(O(n)\)</span></p>
<p>主流排序算法的时间复杂度都是 <span class="arithmatex">\(O(n \log n )\)</span> ,例如快速排序、归并排序、堆排序等。</p>
<p>主流排序算法的时间复杂度通常为 <span class="arithmatex">\(O(n \log n)\)</span> ,例如快速排序、归并排序、堆排序等。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="14:10"><input checked="checked" id="__tabbed_14_1" name="__tabbed_14" type="radio" /><input id="__tabbed_14_2" name="__tabbed_14" type="radio" /><input id="__tabbed_14_3" name="__tabbed_14" type="radio" /><input id="__tabbed_14_4" name="__tabbed_14" type="radio" /><input id="__tabbed_14_5" name="__tabbed_14" type="radio" /><input id="__tabbed_14_6" name="__tabbed_14" type="radio" /><input id="__tabbed_14_7" name="__tabbed_14" type="radio" /><input id="__tabbed_14_8" name="__tabbed_14" type="radio" /><input id="__tabbed_14_9" name="__tabbed_14" type="radio" /><input id="__tabbed_14_10" name="__tabbed_14" type="radio" /><div class="tabbed-labels"><label for="__tabbed_14_1">Java</label><label for="__tabbed_14_2">C++</label><label for="__tabbed_14_3">Python</label><label for="__tabbed_14_4">Go</label><label for="__tabbed_14_5">JavaScript</label><label for="__tabbed_14_6">TypeScript</label><label for="__tabbed_14_7">C</label><label for="__tabbed_14_8">C#</label><label for="__tabbed_14_9">Swift</label><label for="__tabbed_14_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -3929,11 +3926,11 @@ O((n - 1) \frac{n}{2}) = O(n^2)
<p align="center"> Fig. 线性对数阶的时间复杂度 </p>
<h3 id="on_1">阶乘阶 <span class="arithmatex">\(O(n!)\)</span><a class="headerlink" href="#on_1" title="Permanent link">&para;</a></h3>
<p>阶乘阶对应数学上的「全排列」。给定 <span class="arithmatex">\(n\)</span> 个互不重复的元素,求其所有可能的排列方案,方案数量为</p>
<p>阶乘阶对应数学上的「全排列」问题。给定 <span class="arithmatex">\(n\)</span> 个互不重复的元素,求其所有可能的排列方案,方案数量为</p>
<div class="arithmatex">\[
n! = n \times (n - 1) \times (n - 2) \times \cdots \times 2 \times 1
\]</div>
<p>阶乘常使用递归实现。例如以下代码,第一层分裂出 <span class="arithmatex">\(n\)</span> 个,第二层分裂出 <span class="arithmatex">\(n - 1\)</span> 个,…… ,直至<span class="arithmatex">\(n\)</span> 层时终止分裂。</p>
<p>阶乘常使用递归实现。例如以下代码,第一层分裂出 <span class="arithmatex">\(n\)</span> 个,第二层分裂出 <span class="arithmatex">\(n - 1\)</span> 个,以此类推,直至第 <span class="arithmatex">\(n\)</span> 层时终止分裂。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="15:10"><input checked="checked" id="__tabbed_15_1" name="__tabbed_15" type="radio" /><input id="__tabbed_15_2" name="__tabbed_15" type="radio" /><input id="__tabbed_15_3" name="__tabbed_15" type="radio" /><input id="__tabbed_15_4" name="__tabbed_15" type="radio" /><input id="__tabbed_15_5" name="__tabbed_15" type="radio" /><input id="__tabbed_15_6" name="__tabbed_15" type="radio" /><input id="__tabbed_15_7" name="__tabbed_15" type="radio" /><input id="__tabbed_15_8" name="__tabbed_15" type="radio" /><input id="__tabbed_15_9" name="__tabbed_15" type="radio" /><input id="__tabbed_15_10" name="__tabbed_15" type="radio" /><div class="tabbed-labels"><label for="__tabbed_15_1">Java</label><label for="__tabbed_15_2">C++</label><label for="__tabbed_15_3">Python</label><label for="__tabbed_15_4">Go</label><label for="__tabbed_15_5">JavaScript</label><label for="__tabbed_15_6">TypeScript</label><label for="__tabbed_15_7">C</label><label for="__tabbed_15_8">C#</label><label for="__tabbed_15_9">Swift</label><label for="__tabbed_15_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -4068,12 +4065,12 @@ n! = n \times (n - 1) \times (n - 2) \times \cdots \times 2 \times 1
<p align="center"> Fig. 阶乘阶的时间复杂度 </p>
<h2 id="226">2.2.6. &nbsp; 最差、最佳、平均时间复杂度<a class="headerlink" href="#226" title="Permanent link">&para;</a></h2>
<p><strong>某些算法的时间复杂度不是定的,而是与输入数据的分布有关</strong>举一个例子,输入一个长度为 <span class="arithmatex">\(n\)</span> 数组 <code>nums</code> ,其中 <code>nums</code> 由从 <span class="arithmatex">\(1\)</span><span class="arithmatex">\(n\)</span> 的数字组成,但元素顺序是随机打乱的;算法的任务是返回元素 <span class="arithmatex">\(1\)</span> 的索引。我们可以得出以下结论:</p>
<p><strong>某些算法的时间复杂度不是定的,而是与输入数据的分布有关</strong>例如,假设输入一个长度为 <span class="arithmatex">\(n\)</span> 数组 <code>nums</code> ,其中 <code>nums</code> 由从 <span class="arithmatex">\(1\)</span><span class="arithmatex">\(n\)</span> 的数字组成,但元素顺序是随机打乱的;算法的任务是返回元素 <span class="arithmatex">\(1\)</span> 的索引。我们可以得出以下结论:</p>
<ul>
<li><code>nums = [?, ?, ..., 1]</code>,即当末尾元素是 <span class="arithmatex">\(1\)</span> 时,需完整遍历数组,此时达到 <strong>最差时间复杂度 <span class="arithmatex">\(O(n)\)</span></strong> </li>
<li><code>nums = [?, ?, ..., 1]</code> ,即当末尾元素是 <span class="arithmatex">\(1\)</span> 时,需完整遍历数组,此时达到 <strong>最差时间复杂度 <span class="arithmatex">\(O(n)\)</span></strong></li>
<li><code>nums = [1, ?, ?, ...]</code> ,即当首个数字为 <span class="arithmatex">\(1\)</span> 时,无论数组多长都不需要继续遍历,此时达到 <strong>最佳时间复杂度 <span class="arithmatex">\(\Omega(1)\)</span></strong></li>
</ul>
<p>函数渐近上界使用大 <span class="arithmatex">\(O\)</span> 记号表示,代表「最差时间复杂度」。与之对应,「函数渐近下界<span class="arithmatex">\(\Omega\)</span> 记号Omega Notation来表示,代表「最佳时间复杂度」。</p>
<p>函数渐近上界使用大 <span class="arithmatex">\(O\)</span> 记号表示,代表「最差时间复杂度」。相应地,“函数渐近下界<span class="arithmatex">\(\Omega\)</span> 记号来表示,代表「最佳时间复杂度」。</p>
<div class="tabbed-set tabbed-alternate" data-tabs="16:10"><input checked="checked" id="__tabbed_16_1" name="__tabbed_16" type="radio" /><input id="__tabbed_16_2" name="__tabbed_16" type="radio" /><input id="__tabbed_16_3" name="__tabbed_16" type="radio" /><input id="__tabbed_16_4" name="__tabbed_16" type="radio" /><input id="__tabbed_16_5" name="__tabbed_16" type="radio" /><input id="__tabbed_16_6" name="__tabbed_16" type="radio" /><input id="__tabbed_16_7" name="__tabbed_16" type="radio" /><input id="__tabbed_16_8" name="__tabbed_16" type="radio" /><input id="__tabbed_16_9" name="__tabbed_16" type="radio" /><input id="__tabbed_16_10" name="__tabbed_16" type="radio" /><div class="tabbed-labels"><label for="__tabbed_16_1">Java</label><label for="__tabbed_16_2">C++</label><label for="__tabbed_16_3">Python</label><label for="__tabbed_16_4">Go</label><label for="__tabbed_16_5">JavaScript</label><label for="__tabbed_16_6">TypeScript</label><label for="__tabbed_16_7">C</label><label for="__tabbed_16_8">C#</label><label for="__tabbed_16_9">Swift</label><label for="__tabbed_16_10">Zig</label></div>
<div class="tabbed-content">
<div class="tabbed-block">
@ -4337,14 +4334,14 @@ n! = n \times (n - 1) \times (n - 2) \times \cdots \times 2 \times 1
</div>
<div class="admonition tip">
<p class="admonition-title">Tip</p>
<p>我们在实际应用中很少使用「最佳时间复杂度」,因为往往只有很小概率下才能达到,会带来一定的误导性。反,「最差时间复杂度」为实用,因为它给出了一个“效率安全值”,让我们可以放心地使用算法。</p>
<p>实际应用中我们很少使用「最佳时间复杂度」,因为通常只有很小概率下才能达到,可能会带来一定的误导性。反,「最差时间复杂度」为实用,因为它给出了一个“效率安全值”,让我们可以放心地使用算法。</p>
</div>
<p>从上述示例可以看出,最差或最佳时间复杂度只出现在“特殊分布的数据”中,这些情况的出现概率往往很小,因此并不能最真实地反映算法运行效率。<strong>相对地,「平均时间复杂度」可以体现算法在随机输入数据下的运行效率,用 <span class="arithmatex">\(\Theta\)</span> 记号Theta Notation来表示</strong></p>
<p>从上述示例可以看出,最差或最佳时间复杂度只出现在“特殊分布的数据”中,这些情况的出现概率可能很小,因此并不能最真实地反映算法运行效率。相较之下,<strong>「平均时间复杂度」可以体现算法在随机输入数据下的运行效率</strong>,用 <span class="arithmatex">\(\Theta\)</span> 记号来表示</p>
<p>对于部分算法,我们可以简单地推算出随机数据分布下的平均情况。比如上述示例,由于输入数组是被打乱的,因此元素 <span class="arithmatex">\(1\)</span> 出现在任意索引的概率都是相等的,那么算法的平均循环次数则是数组长度的一半 <span class="arithmatex">\(\frac{n}{2}\)</span> ,平均时间复杂度为 <span class="arithmatex">\(\Theta(\frac{n}{2}) = \Theta(n)\)</span></p>
<p>但在实际应用中,尤其是较为复杂的算法,计算平均时间复杂度比较困难,因为很难简便地分析出在数据分布下的整体数学期望。这种情况下,我们一般使用最差时间复杂度作为算法效率的评判标准。</p>
<p>但在实际应用中,尤其是较为复杂的算法,计算平均时间复杂度比较困难,因为很难简便地分析出在数据分布下的整体数学期望。这种情况下,我们通常使用最差时间复杂度作为算法效率的评判标准。</p>
<div class="admonition question">
<p class="admonition-title">为什么很少看到 <span class="arithmatex">\(\Theta\)</span> 符号?</p>
<p>实际中我们经常使用「大 <span class="arithmatex">\(O\)</span> 符号来表示「平均复杂度」,这样严格意义上来说是不规范的。这可能是因为 <span class="arithmatex">\(O\)</span> 符号实在是太朗朗上口了。</br>如果在本书和其他资料中看到类似 <strong>平均时间复杂度 <span class="arithmatex">\(O(n)\)</span></strong> 的表述,请直接理解为 <span class="arithmatex">\(\Theta(n)\)</span> 即可</p>
<p>可能由于 <span class="arithmatex">\(O\)</span> 符号过于朗朗上口,我们常常使用它来表示「平均复杂度」,但从严格意义上看,这种做法并不规范。在本书和其他资料中,若遇到类似“平均时间复杂度 <span class="arithmatex">\(O(n)\)</span>的表述,请将其直接理解为 <span class="arithmatex">\(\Theta(n)\)</span></p>
</div>

View File

@ -2166,7 +2166,7 @@
<p><strong>空间复杂度 <span class="arithmatex">\(O(n + m)\)</span></strong> :借助了长度分别为 <span class="arithmatex">\(n\)</span> , <span class="arithmatex">\(m\)</span> 的数组 <code>res</code><code>counter</code> ,是“非原地排序”。</p>
<p><strong>稳定排序</strong>:由于向 <code>res</code> 中填充元素的顺序是“从右向左”的,因此倒序遍历 <code>nums</code> 可以避免改变相等元素之间的相对位置,从而实现“稳定排序”;其实正序遍历 <code>nums</code> 也可以得到正确的排序结果,但结果“非稳定”。</p>
<h2 id="1174">11.7.4. &nbsp; 局限性<a class="headerlink" href="#1174" title="Permanent link">&para;</a></h2>
<p>看到这里,你也许会觉得计数排序太妙了,咔咔一通操作,时间复杂度就下来了。然而,使用技术排序的前置条件比较苛刻。</p>
<p>看到这里,你也许会觉得计数排序太妙了,咔咔一通操作,时间复杂度就下来了。然而,使用计数排序的前置条件比较苛刻。</p>
<p><strong>计数排序只适用于非负整数</strong>。若想要用在其他类型数据上,则要求该数据必须可以被转化为非负整数,并且不能改变各个元素之间的相对大小关系。例如,对于包含负数的整数数组,可以先给所有数字加上一个常数,将全部数字转化为正数,排序完成后再转换回去即可。</p>
<p><strong>计数排序适用于数据量大但数据范围不大的情况</strong>。比如,上述示例中 <span class="arithmatex">\(m\)</span> 不能太大,否则占用空间太多;而当 <span class="arithmatex">\(n \ll m\)</span> 时,计数排序使用 <span class="arithmatex">\(O(m)\)</span> 时间,有可能比 <span class="arithmatex">\(O(n \log n)\)</span> 的排序算法还要慢。</p>

File diff suppressed because one or more lines are too long

View File

@ -2,272 +2,272 @@
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
<url>
<loc>https://www.hello-algo.com/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_appendix/contribution/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_appendix/installation/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_array_and_linkedlist/array/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_array_and_linkedlist/linked_list/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_array_and_linkedlist/list/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_array_and_linkedlist/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_computational_complexity/performance_evaluation/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_computational_complexity/space_complexity/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_computational_complexity/space_time_tradeoff/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_computational_complexity/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_computational_complexity/time_complexity/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_data_structure/classification_of_data_structure/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_data_structure/data_and_memory/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_data_structure/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_graph/graph/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_graph/graph_operations/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_graph/graph_traversal/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_graph/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_hashing/hash_collision/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_hashing/hash_map/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_hashing/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_heap/build_heap/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_heap/heap/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_heap/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_introduction/algorithms_are_everywhere/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_introduction/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_introduction/what_is_dsa/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_preface/about_the_book/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_preface/suggestions/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_preface/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_reference/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_searching/binary_search/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_searching/hashing_search/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_searching/linear_search/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_searching/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/bubble_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/bucket_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/counting_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/insertion_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/intro_to_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/merge_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/quick_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/radix_sort/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_sorting/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_stack_and_queue/deque/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_stack_and_queue/queue/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_stack_and_queue/stack/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_stack_and_queue/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_tree/avl_tree/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_tree/binary_search_tree/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_tree/binary_tree/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_tree/binary_tree_traversal/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
<url>
<loc>https://www.hello-algo.com/chapter_tree/summary/</loc>
<lastmod>2023-04-06</lastmod>
<lastmod>2023-04-07</lastmod>
<changefreq>daily</changefreq>
</url>
</urlset>

Binary file not shown.