This commit is contained in:
krahets
2023-05-24 00:32:47 +08:00
parent 9a0252f484
commit c6937e63b7
80 changed files with 4101 additions and 905 deletions

View File

@ -25,7 +25,7 @@
<title>11.7.   计数排序 - Hello 算法</title>
<title>11.8.   计数排序 - Hello 算法</title>
@ -79,7 +79,7 @@
<div data-md-component="skip">
<a href="#117" class="md-skip">
<a href="#118" class="md-skip">
跳转至
</a>
@ -113,7 +113,7 @@
<div class="md-header__topic" data-md-component="header-topic">
<span class="md-ellipsis">
11.7. &nbsp; 计数排序
11.8. &nbsp; 计数排序
</span>
</div>
@ -1409,6 +1409,8 @@
@ -1445,9 +1447,23 @@
<li class="md-nav__item">
<a href="../selection_sort/" class="md-nav__link">
11.2. &nbsp; 选择排序
</a>
</li>
<li class="md-nav__item">
<a href="../bubble_sort/" class="md-nav__link">
11.2. &nbsp; 冒泡排序
11.3. &nbsp; 冒泡排序
</a>
</li>
@ -1461,7 +1477,7 @@
<li class="md-nav__item">
<a href="../insertion_sort/" class="md-nav__link">
11.3. &nbsp; 插入排序
11.4. &nbsp; 插入排序
</a>
</li>
@ -1475,7 +1491,7 @@
<li class="md-nav__item">
<a href="../quick_sort/" class="md-nav__link">
11.4. &nbsp; 快速排序
11.5. &nbsp; 快速排序
</a>
</li>
@ -1489,7 +1505,7 @@
<li class="md-nav__item">
<a href="../merge_sort/" class="md-nav__link">
11.5. &nbsp; 归并排序
11.6. &nbsp; 归并排序
</a>
</li>
@ -1503,7 +1519,7 @@
<li class="md-nav__item">
<a href="../bucket_sort/" class="md-nav__link">
11.6. &nbsp; 桶排序
11.7. &nbsp; 桶排序
</a>
</li>
@ -1526,12 +1542,12 @@
<label class="md-nav__link md-nav__link--active" for="__toc">
11.7. &nbsp; 计数排序
11.8. &nbsp; 计数排序
<span class="md-nav__icon md-icon"></span>
</label>
<a href="./" class="md-nav__link md-nav__link--active">
11.7. &nbsp; 计数排序
11.8. &nbsp; 计数排序
</a>
@ -1550,29 +1566,29 @@
<ul class="md-nav__list" data-md-component="toc" data-md-scrollfix>
<li class="md-nav__item">
<a href="#1171" class="md-nav__link">
11.7.1. &nbsp; 简单实现
<a href="#1181" class="md-nav__link">
11.8.1. &nbsp; 简单实现
</a>
</li>
<li class="md-nav__item">
<a href="#1172" class="md-nav__link">
11.7.2. &nbsp; 完整实现
<a href="#1182" class="md-nav__link">
11.8.2. &nbsp; 完整实现
</a>
</li>
<li class="md-nav__item">
<a href="#1173" class="md-nav__link">
11.7.3. &nbsp; 算法特性
<a href="#1183" class="md-nav__link">
11.8.3. &nbsp; 算法特性
</a>
</li>
<li class="md-nav__item">
<a href="#1174" class="md-nav__link">
11.7.4. &nbsp; 局限性
<a href="#1184" class="md-nav__link">
11.8.4. &nbsp; 局限性
</a>
</li>
@ -1593,7 +1609,7 @@
<li class="md-nav__item">
<a href="../radix_sort/" class="md-nav__link">
11.8. &nbsp; 基数排序
11.9. &nbsp; 基数排序
</a>
</li>
@ -1607,7 +1623,7 @@
<li class="md-nav__item">
<a href="../summary/" class="md-nav__link">
11.9. &nbsp; 小结
11.10. &nbsp; 小结
</a>
</li>
@ -1852,29 +1868,29 @@
<ul class="md-nav__list" data-md-component="toc" data-md-scrollfix>
<li class="md-nav__item">
<a href="#1171" class="md-nav__link">
11.7.1. &nbsp; 简单实现
<a href="#1181" class="md-nav__link">
11.8.1. &nbsp; 简单实现
</a>
</li>
<li class="md-nav__item">
<a href="#1172" class="md-nav__link">
11.7.2. &nbsp; 完整实现
<a href="#1182" class="md-nav__link">
11.8.2. &nbsp; 完整实现
</a>
</li>
<li class="md-nav__item">
<a href="#1173" class="md-nav__link">
11.7.3. &nbsp; 算法特性
<a href="#1183" class="md-nav__link">
11.8.3. &nbsp; 算法特性
</a>
</li>
<li class="md-nav__item">
<a href="#1174" class="md-nav__link">
11.7.4. &nbsp; 局限性
<a href="#1184" class="md-nav__link">
11.8.4. &nbsp; 局限性
</a>
</li>
@ -1902,9 +1918,9 @@
<h1 id="117">11.7. &nbsp; 计数排序<a class="headerlink" href="#117" title="Permanent link">&para;</a></h1>
<h1 id="118">11.8. &nbsp; 计数排序<a class="headerlink" href="#118" title="Permanent link">&para;</a></h1>
<p>「计数排序 Counting Sort」通过统计元素数量来实现排序通常应用于整数数组。</p>
<h2 id="1171">11.7.1. &nbsp; 简单实现<a class="headerlink" href="#1171" title="Permanent link">&para;</a></h2>
<h2 id="1181">11.8.1. &nbsp; 简单实现<a class="headerlink" href="#1181" title="Permanent link">&para;</a></h2>
<p>先来看一个简单的例子。给定一个长度为 <span class="arithmatex">\(n\)</span> 的数组 <code>nums</code> ,其中的元素都是“非负整数”。计数排序的整体流程如下:</p>
<ol>
<li>遍历数组,找出数组中的最大数字,记为 <span class="arithmatex">\(m\)</span> ,然后创建一个长度为 <span class="arithmatex">\(m + 1\)</span> 的辅助数组 <code>counter</code> </li>
@ -2149,7 +2165,7 @@
<p class="admonition-title">计数排序与桶排序的联系</p>
<p>从桶排序的角度看,我们可以将计数排序中的计数数组 <code>counter</code> 的每个索引视为一个桶,将统计数量的过程看作是将各个元素分配到对应的桶中。本质上,计数排序是桶排序在整型数据下的一个特例。</p>
</div>
<h2 id="1172">11.7.2. &nbsp; 完整实现<a class="headerlink" href="#1172" title="Permanent link">&para;</a></h2>
<h2 id="1182">11.8.2. &nbsp; 完整实现<a class="headerlink" href="#1182" title="Permanent link">&para;</a></h2>
<p>细心的同学可能发现,<strong>如果输入数据是对象,上述步骤 <code>3.</code> 就失效了</strong>。例如,输入数据是商品对象,我们想要按照商品价格(类的成员变量)对商品进行排序,而上述算法只能给出价格的排序结果。</p>
<p>那么如何才能得到原数据的排序结果呢?我们首先计算 <code>counter</code> 的「前缀和」。顾名思义,索引 <code>i</code> 处的前缀和 <code>prefix[i]</code> 等于数组前 <code>i</code> 个元素之和,即</p>
<div class="arithmatex">\[
@ -2509,11 +2525,13 @@
</div>
</div>
</div>
<h2 id="1173">11.7.3. &nbsp; 算法特性<a class="headerlink" href="#1173" title="Permanent link">&para;</a></h2>
<p><strong>时间复杂度 <span class="arithmatex">\(O(n + m)\)</span></strong> :涉及遍历 <code>nums</code> 和遍历 <code>counter</code> ,都使用线性时间。一般情况下 <span class="arithmatex">\(n \gg m\)</span> ,时间复杂度趋于 <span class="arithmatex">\(O(n)\)</span></p>
<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>
<h2 id="1183">11.8.3. &nbsp; 算法特性<a class="headerlink" href="#1183" title="Permanent link">&para;</a></h2>
<ul>
<li><strong>间复杂度 <span class="arithmatex">\(O(n + m)\)</span></strong> 涉及遍历 <code>nums</code> 和遍历 <code>counter</code> ,都使用线性时间。一般情况下 <span class="arithmatex">\(n \gg m\)</span> ,时间复杂度趋于 <span class="arithmatex">\(O(n)\)</span></li>
<li><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></li>
<li><strong>稳定排序</strong>:由于向 <code>res</code> 中填充元素的顺序是“从右向左”的,因此倒序遍历 <code>nums</code> 可以避免改变相等元素之间的相对位置,从而实现稳定排序。实际上,正序遍历 <code>nums</code> 也可以得到正确的排序结果,但结果是非稳定的。</li>
</ul>
<h2 id="1184">11.8.4. &nbsp; 局限性<a class="headerlink" href="#1184" title="Permanent link">&para;</a></h2>
<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>
@ -2594,7 +2612,7 @@
<nav class="md-footer__inner md-grid" aria-label="页脚" >
<a href="../bucket_sort/" class="md-footer__link md-footer__link--prev" aria-label="上一页: 11.6. &amp;nbsp; 桶排序" rel="prev">
<a href="../bucket_sort/" class="md-footer__link md-footer__link--prev" aria-label="上一页: 11.7. &amp;nbsp; 桶排序" rel="prev">
<div class="md-footer__button md-icon">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M20 11v2H8l5.5 5.5-1.42 1.42L4.16 12l7.92-7.92L13.5 5.5 8 11h12Z"/></svg>
</div>
@ -2603,20 +2621,20 @@
上一页
</span>
<div class="md-ellipsis">
11.6. &nbsp; 桶排序
11.7. &nbsp; 桶排序
</div>
</div>
</a>
<a href="../radix_sort/" class="md-footer__link md-footer__link--next" aria-label="下一页: 11.8. &amp;nbsp; 基数排序" rel="next">
<a href="../radix_sort/" class="md-footer__link md-footer__link--next" aria-label="下一页: 11.9. &amp;nbsp; 基数排序" rel="next">
<div class="md-footer__title">
<span class="md-footer__direction">
下一页
</span>
<div class="md-ellipsis">
11.8. &nbsp; 基数排序
11.9. &nbsp; 基数排序
</div>
</div>
<div class="md-footer__button md-icon">