第7章 排序

news2025/1/15 19:57:14

前言

        在这一章,我们讨论数组元素的排序问题。为简单起见,假设在我们的例子中数组只包含整数,虽然更复杂的结构显然也是可能的。对于本章的大部分内容,我们还假设整个排序工作能够在主存中完成,因此,元素的个数相对来说比较小(小于\mathit{10^{6}})。当然,不能在主存中完成而必须在磁盘或磁带上完成的排序也相当重要。这种类型的排序叫作外部排序(external sorting),将在本章末尾讨论外部排序。

        我们对内部排序的考察将指出:

  • 存在几种容易的算法以\mathit{O(N^{2})}排序,如插入排序。
  • 有一种算法叫作希尔排序(Shellsort),它的编程非常简单,以\mathit{o(N^{2})}运行,并在实践中很有效。
  • 有一些稍微复杂的\mathit{O(NlogN)}的排序算法。
  • 任何通用的排序算法均需要\mathit{\Omega (NlogN)}次比较。

        本章的其余部分将描述和分析各种排序算法。这些算法包含一些有趣的、重要的代码优化和算法设计思想。可以对排序做出精确的分析。预先说明,在适当的时候,我们将尽可能地多做一些分析。

7.1 预备知识

        我们描述的算法都将是可以互换的。每个算法都将接收一个含有元素的数组和一个包
含元素个数的整数。

        我们将假设\mathit{N}是传递到排序例程中的元素个数,它已经被检查过,是合法的。按照C
的约定,对于所有的排序,数据都将在位置0处开始。

        我们还假设“<”和“>”运算符存在,它们可以用于对输入进行一致的排序。除赋
值运算符外,这两种运算是仅有的允许对输入数据进行的操作。在这些条件下的排序叫作
基于比较的排序(comparison-based sorting)。

7.2 插入排序

7.2.1 算法

        最简单的排序算法之一是插入排序(insertion sort)。插入排序由\mathit{N-1}趟(pass)排序组成。对于\mathit{P=1}趟到\mathit{P=N-1}趟,插入排序保证从位置0到位置\mathit{P}上的元素为已排序状态。插入排序利用了这样的事实:位置0到位置\mathit{P-1}上的元素是已排过序的。图7-1显示一个简单的数组在每一趟插入排序后的情况。

        图7-1表达了一般的方法。在第\mathit{P}趟,我们将位置\mathit{P}上的元素向左移动到它在前\mathit{P+1}个元素中的正确位置上。图7-2中的程序实现该想法。第2~5行实现数据移动而没有明显使用交换。将位置\mathit{P}上的元素存于Tmp中,而(在位置\mathit{P}之前)所有更大的元素都向右移动一个位置。然后将Tmp置于正确的位置上。这种方法与实现二叉堆时所用到的技巧相同。

void InsertionSort(ElementType A[], int N)
{
    int j, P;
    
    ElementType Tmp;
    for (P = 1; P < N; P++)
    {
        Tmp = A[P];
        for (j = P; j > 0 && A[j - 1] > Tmp; j--)
            A[j] = A[j - 1];
        A[j] = Tmp;
    }
}

7.2.2 插入排序的分析

        由于嵌套循环每趟花费N次迭代,因此插入排序为\mathit{O(N^{2})},而且这个界是精确的,因为以反序输入可以达到该界。精确计算指出对于\mathit{P}的每一个值,第4行的测试最多执行\mathit{P+1}次。对所有的\mathit{P}求和,得到总数为

\sum_{i=2}^{N}i=2+3+4+\cdot \cdot \cdot +N=\Theta (N^{2})

        另一方面,如果输入数据已预先排序,那么运行时间为\mathit{O(N)},因为内层for循环的检测总是立即判定不成立而终止。事实上,如果输入几乎已排序(该术语将在下一节更严格地定义),那么插入排序将运行得很快。由于这种变化差别很大,因此值得我们去分析该算法平均情形的行为。实际上,和各种其他排序算法一样,插入排序的平均情形也是\Theta (N^{2}),详见下节的分析。

7.3 一些简单排序算法的下界

        数字数组的一个逆序(inversion)是指数组中具有\mathit{i<j}\mathit{A[i]>A[j]}的序偶(\mathit{A[i],A[j]})。在上节的例子中,输入数据34,8,64,51,32,21有9个逆序,即(34,8),(34,32),(34,21),(64,51),(64,32),(64,21),(51,32),(51,21),(32,21)。这正好是需要由插入排序(非直接)执行的交换次数。情况总是这样,因为交换两个不按原序排列的相邻元素恰好消除一个逆序,而一个排过序的数组没有逆序。由于算法中还有\mathit{O(N)}项其他的工作,因此插入排序的运行时间是\mathit{O(I+N)},其中\mathit{I}为原始数组中的逆序数。于是,若逆序数是\mathit{O(N)},则插入排序以线性时间运行。

        我们可以通过计算排列中的平均逆序数而得出插入排序平均运行时间的精确的界。如往常一样,定义平均是一个困难的命题。我们将假设不存在重复元素(如果允许重复,那么我们甚至连重复的平均次数究竟是什么都不清楚)。利用该假设,我们可设输入数据是前\mathit{N}个整数的某个排列(因为只有相对顺序才是重要的),并设所有的排列都是等可能的。在这些假设下,我们有如下定理:

        定理 7.1 \mathit{N}个互异数的数组的平均逆序数是\mathit{N(N-1)/4}

        证明:对于含有任意的数的表\mathit{L},考虑其反序表\mathit{L_{r}}。上例中的反序表是21,32,51,64,8,34。考虑该表中任意两个数的序偶(x,y),且y>x。显然,恰是\mathit{L}\mathit{L_{r}}之中的一个,该序偶表示一个逆序。在表\mathit{L}和它的反序表\mathit{L_{r}},中序偶的总个数为\mathit{N(N-1)/2}。因此,平均表有该量的一半,即\mathit{N(N-1)/4}个逆序。

        这个定理意味着插入排序平均是二次的,同时也提供了只交换相邻元素的任何算法的一个很强的下界。

        定理 7.2 通过交换相邻元素进行排序的任何算法平均需要\mathit{\Omega (N^{2})}时间。

        证明:初始的平均逆序数 是\mathit{N(N-1)/4=\Omega (N^{2})},而每次交换只减少一个逆序,因此需要\mathit{\Omega (N^{2})}次交换。

        这是证明下界的一个例子,它不仅对非显式地实施相邻元素的交换的插入排序有效,而且对诸如冒泡排序和选择排序等其他一些简单算法也是有效的,不过这些算法将不在这里描述。事实上,它对一整类只进行相邻元素的交换的排序算法(包括那些未被发现的算法)都是有效的。正因为如此,这个证明在经验上是不能被认可的。虽然这个下界的证明非常简单,但是一般说来证明下界要比证明上界复杂得多。

        这个下界告诉我们,为了使一个排序算法以亚二次(subquadratic)或\mathit{o (N^{2})}时间运行,必须执行一些比较,特别要对相距较远的元素进行交换。一个排序算法通过删除逆序得以向前进行,而为了有效地运行,它必须每次交换删除不止一个逆序。

7.4 希尔排序

        希尔排序(Shellsort)的名称源于它的发明者Donald Shell,该算法是冲破二次时间屏障的第一批算法之一,不过,从它的发现之日起,又过了若干年后才证明了它的亚二次时间界。正如上节所提到的,它通过比较相距一定间隔的元素来工作,各趟比较所用的距离随着算法的进行而减小,直到只比较相邻元素的最后一趟排序为止。由于这个原因,希尔排序有时也叫作缩小增量排序(diminishing increment sort)。

        希尔排序使用一个序列\mathit{h_{1},h_{2},\cdot \cdot \cdot ,h_{t}}叫作增量序列(increment sequence)。只要\mathit{h_{1}=1},任何增量序列都是可行的,不过,有些增量序列比另外一些增量序列更好(后面我们将讨论这个问题)。在使用增量\mathit{h_{k}}的一趟排序之后,对于每一个\mathit{i}我们有\mathit{A[i]\leqslant A[i+h_{k}]}(这里它是有意义的),所有相隔\mathit{h_{k}}的元素都被排序。此时称文件是\mathit{h_{k}}-排序的(\mathit{h_{k}}-sorted)。例如,图7-3显示了各趟排序后数组的情况。希尔排序的一个重要性质(我们只叙述而不证明)是一个\mathit{h_{k}}-排序的文件(此后将是\mathit{h_{k-1}}-排序的)保持它的\mathit{h_{k}}-排序性。事实上,假如情况不是这样的话,那么该算法也就没什么意义了,因为前面各趟排序的结果会被后面各趟排序给打乱。

        \mathit{h_{k}}-排序的一般做法是,对于\mathit{h_{k},h_{k}+1,\cdot \cdot \cdot ,N-1}中的每一个位置\mathit{i},把其上的元素放到\mathit{i,i-h_{k},i-2h_{k}\cdot \cdot \cdot }中间的正确位置上。虽然这并不影响最终结果,但是仔细的考察指出,一趟\mathit{h_{k}}-排序的作用就是对\mathit{h_{k}}个独立的子数组执行一次插入排序。当我们分析希尔排序的运行时间时,这个考察结果将是很重要的。

        增量序列的一种流行(但是不好)的选择是使用Shell建议的序列:\mathit{h_{t}=[N/2]}\mathit{h_{k}=[h_{k+1}/2]}图7-4包含一个使用该序列实现希尔排序的程序。后面我们将看到,存在一些递增的序列,它们对该算法的运行时间做出了重要的改进,即使是一个小的改变都可能剧烈地影响算法的性能。

void ShellSort(ElementType A[], int N)
{
    int i, j, Increment;
    ElementType Tmp;

    for (Increment = N / 2; Increment > 0; Increment /= 2)
        for(i = Increment; i < N; i++)
        {
            Tmp = A[i];
            for (j = i; j >= Increment; j -= Increment)
                if(Tmp < A[j - Increment])
                    A[j] = A[j - Increment];
                else
                    break;
            A[j] = Tmp;
        }
}

希尔排序的最坏情形分析

        虽然希尔排序编程简单,但是,其运行时间的分析则完全是另外一回事。希尔排序的运行时间依赖于增量序列的选择,而证明可能相当复杂。希尔排序的平均情形分析,除最平凡的一些增量序列外,是一个长期未解决的问题。我们将证明在两个特别的增量序列下最坏情形的精确的界。

        定理 7.3 使用希尔增量时希尔排序的最坏情形运行时间为\Theta (N^{2})

        证明:证明不仅需要指出最坏情形运行时间的上界,而且还需要指出存在某个输入实际上就花费\Omega (N^{2})时间运行。首先通过构造一个坏情形来证明下界。我们先选择\mathit{N}是2的幂。这使得除最后一个增量是1外所有的增量都是偶数。现在,我们给出一个数组Input-Data作为输入,它的偶数位置上有\mathit{N/2}个同为最大的数,而在奇数位置上有\mathit{N/2}个同为最小的数(对该证明,第一个位置是位置1)。由于除最后一个增量外所有的增量都是偶数,因此,当我们进行最后一趟排序前,\mathit{N/2}个最大的元素仍然处在偶数位置上,而\mathit{N/2}个最小的元素也还是在奇数位置上。于是,在最后一趟排序开始之前第\mathit{i}个最小的数(\mathit{i\leqslant N/2})在位置\mathit{2i-1}上。将第\mathit{i}个元素恢复到其正确位置需要在数组中移动\mathit{i-1}个间隔。这样,仅仅将\mathit{N/2}个最小的元素放到正确的位置上就至少需要\sum_{i=1}^{N/2}i-1=\Omega (N^{2})的工作。举一个例子,图7-5显示一个\mathit{N=16}时的坏(但不是最坏)的输入。在2-排序后的逆序数一直恰好保持为1+2+3+4+5+6+7=28,因此,最后一趟排序将花费相当多的时间。

        现在我们证明上界\mathit{O(N^{2})}以结束本证明。前面已经观察到,带有增量\mathit{h_{k}}的一趟排序由\mathit{h_{k}}个关于\mathit{N/h_{k}}个元素的插入排序组成。由于插入排序是二次的,因此一趟排序总的开销是\mathit{O(h_{k}(N/h_{k})^{2})=O(N^{2}/h_{k})}。对所有各趟排序求和则给出总的界为\mathit{O(\sum_{i=1}^{t}N^{2}/h_{i})=O(N^{2}\sum_{i=1}^{t}1/h_{k})}。因为这些增量形成一个几何级数,其公比为2,而该级数中的最大项是
\mathit{h_{1}=1},因此,\mathit{\sum_{i=1}^{t}1/h_{i}<2}。于是,我们得到总的界\mathit{O(N^{2})}

        希尔增量的问题在于,这些增量对未必互素,因此较小的增量可能影响很小。Hibbard提出一个稍微不同的增量序列,它在实践中(并且理论上)给出更好的结果。他的增量形如\mathit{1,3,7,\cdot \cdot \cdot ,2^{k}-1}。虽然这些增量几乎是相同的,但关键的区别是相邻的增量没有公因子。现在我们就来分析使用这个增量序列的希尔排序的最坏情形运行时间,这个证明相当复杂。

        定理 7.4 使用Hibbard增量的希尔排序的最坏情形运行时间为\mathit{\Theta (N^{3/2})}

        证明:我们只证明上界而将下界的证明留作练习。这个证明需要堆垒数论(additivenumber theory)中某些众所周知的结果。本章末提供了这些结果的参考资料。

        和前面一样,对于上界,我们还是计算每一趟排序的运行时间的界,然后对各趟求和。对于那些\mathit{h_{k}>N^{1/2}}的增量,我们将使用前一定理得到的界\mathit{O(N^{2}/h_{k})}。虽然这个界对于其他增量也是成立的,但是它太大,用不上。直观地看,我们必须利用这个增量序列是特殊的这样一个事实。我们需要证明的是,对于位置\mathit{P}上的任意元素\mathit{A_{p}},当要执行\mathit{h_{k}}-排序时,只有少数元素在位置\mathit{P}的左边且大于\mathit{A_{p}}

        当对输入数组进行\mathit{h_{k}}-排序时,我们知道它已经是\mathit{h_{k+1}}-排序和\mathit{h_{k+2}}-排序的了。在\mathit{h_{k}}-排序以前,考虑位置\mathit{P}\mathit{P-i}上的两个元素,其中\mathit{i\leqslant P}。如果\mathit{i}\mathit{h_{k+1}}\mathit{h_{k+2}}的倍数,那么显然\mathit{A[P-i]<A[P]}。不仅如此,如果\mathit{i}可以表示为\mathit{h_{k+1}}\mathit{h_{k+2}}的线性组合(以非负整数的形式),那么也有\mathit{A[P-i]<A[P]}。例如,当我们进行3-排序时,文件已经是7-排序和
15-排序的了。52可以表示为7和15的线性组合:52=1×7+3×15。因此,A[100]不可能大于A[152],因为\mathit{A[100]\leqslant A[107]\leqslant A[122]\leqslant A[137]\leqslant A[152]}

        现在,\mathit{h_{k+2}=2h_{k+1}+1},因此\mathit{h_{k+1}}\mathit{h_{k+2}}没有公因子。在这种情形下,可以证明,至少和\mathit{(h_{k+1}-1)(h_{k+2}-1)=8h_{k}^{2}+4h_{k}}一样大的所有整数都可以表示为\mathit{h_{k+1}}\mathit{h_{k+2}}的线性组合(见本章末尾的参考文献)。

        这就告诉我们,第4行的for循环体对于这些\mathit{N-h_{k}}位置上的每一个,最多执行\mathit{8h_{k}+4=O(h_{k})}次。于是我们得到每趟的界\mathit{O(Nh_{k})}

        利用大约一半的增量满足\mathit{h_{k}<\sqrt{N}}的事实并假设\mathit{t}是偶数,那么总的运行时间为
\mathit{O(\sum_{k+1}^{t/2}Nh_{k}+\sum_{k=t/2+1}^{t}N^{2}/h_{k})=O(N\sum_{k=1}^{t/2}h_{k}+N^{2}\sum_{k=t/2+1}^{t}1/h_{k})}
因为两个和都是几何级数,并且\mathit{h_{t/2}=\Theta (\sqrt{N})},所以上式简化为
\mathit{=O(Nh_{t/2})+O(\frac{N^{2}}{h_{t/2}})=O(N^{3/2})}

        使用Hibbard增量的希尔排序平均情形运行时间基于模拟的结果被认为是\mathit{O(N^{5/4})},但是没有人能够证明该结果。Pratt已经证明,\mathit{\Theta (N^{3/2})}的界适用于广泛的增量序列。

        Sedgewick 提出了几种增量序列,其最坏情形运行时间(也是可以达到的)为\mathit{O(N^{4/3})}。对于这些增量序列的平均运行时间猜测为\mathit{O(N^{7/6})}。经验研究指出,在实践中这些序列的运行要比Hibbard的好得多,其中最好的是序列(1,5,19,41,109,...),该序列中的项或者是\mathit{9\cdot 4^{i}-9\cdot 2^{i}+1},或者是\mathit{4^{i}-3\cdot 2^{i}+1}。通过将这些值放到一个数组中可以最容易地实现该算法。虽然有可能存在某个增量序列使得能够对希尔排序的运行时间做出重大改进,但是,这个增量在实践中还是最为人们称道的。

        关于希尔排序还有几个其他结果,它们需要数论和组合数学中一些艰深的定理而且主要是在理论上有用。希尔排序是算法非常简单又具有极其复杂的分析的一个好例子。

        希尔排序的性能在实践中是完全可以接受的,即使是对于数以万计的\mathit{N}仍是如此。编程的简单特点使得它成为对较大的输入数据经常选用的算法。

7.5 堆排序

7.6 归并排序

7.7 快速排序

7.7.1 选取枢纽元

7.7.2 分割策略

7.7.3 小数组

7.7.4 实际的快速排序例程

7.7.5 快速排序的分析

7.7.6 选择的线性期望时间算法

7.8 大型结构的排序

7.9 排序的一般下界

7.10 桶式排序

7.11 外部排序

7.11.1 为什么需要新的算法

7.11.2 外部排序模型

7.11.3 简单算法

7.11.4 多路合并

7.11.5 多相合并

7.11.6 替换选择

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1318668.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Redis Cluster集群搭建 三主三从

Redis包下载 Linux&#xff1a; http://download.redis.io/releases/ Mac or Windows: https://redis.io/download/ 2.下载后解压进入文件夹&#xff08;本次我的Redis版本是6.2.14版本&#xff09; /redis/redis-6.2.14 开始安装 make instarll修改配置文件复制redis.conf 6…

pve8.1版本安装及环境初始化过程记录

背景 经历pve旧版本奔溃事件&#xff0c;果断重新购买了装备&#xff0c;做个稳定的pve环境&#xff0c;旧主机用于折腾其他系统&#xff0c;硬盘一定要用好的。 1.pve8.1.3系统版本安装 下载地址 Download Proxmox software, datasheets, agreements 使用种子下载会快一点儿…

英伟达盒子 Jetson Xshell连接串口查看日志方法(串口日志、盒子日志)

文章目录 连接串口xshell连接串口信息 连接串口 接盒子上的A2、B2&#xff0c;以及接地线&#xff1a; 另外一头接上电脑&#xff08;我用的RS485转USB工具&#xff09;&#xff1a; xshell连接 协议选择SERIAL&#xff1a; 设置盒子厂商约定的端口号、波特率、数据位、停止位…

ElaticSearch 是如何建立索引的?

前面讲到了 NoSQL 数据库的应用&#xff0c;在关系型数据库和 NoSQL 数据库之外&#xff0c;还有一类非常重要的存储中间件&#xff0c;那就是文件索引。当你在电商网站搜索商品&#xff0c;或者在搜索引擎搜索资料时&#xff0c;都离不开基于文件索引的各种检索框架的支持。 …

十二星座对音乐有感觉但是唱得最不着调的排名。

第一名&#xff08;狮子座&#xff09;、第二名&#xff08;水瓶座&#xff09;、第三名&#xff08;射手座&#xff09; 第四名&#xff08;金牛座&#xff09;、第五名&#xff08;摩羯座&#xff09;、第六名&#xff08;天秤座&#xff09; 第七名&#xff08;巨蟹座&#…

03 使用Vite开发Vue3项目

概述 要使用vite创建Vue3项目&#xff0c;有很多种方式&#xff0c;如果使用命令&#xff0c;则推荐如下命令&#xff1a; # 使用nvm将nodejs的版本切换到20 nvm use 20# 全局安装yarn npm install -g yarn# 使用yarnvite创建项目 yarn create vite不过&#xff0c;笔者更推荐…

解决“bat中文路径乱码“问题

今天&#xff0c;在使用.bat脚本&#xff0c;将hello.png从"D:\mypic\备份"目录&#xff0c;拷贝到"D:\mypic\备份"时&#xff1b;发现中文乱码,弹出如下对话框: 图(1) bat中文路径乱码 原来的命令是&#xff1a; copy D:\mypic\one\hello.png D:\mypic\备…

WordPress VIP收费下载插件Erphpdown v17.0.1 开心版

会员推广下载专业版 WordPress插件&#xff08;erphpdown&#xff09;是模板兔开发的一款针对虚拟资源收费下载/付费下载/付费视频/收费查看/付费阅读/付费查看/VIP下载查看的插件&#xff0c;经过完美测试运行于wordpress 3.x-5.x版本。后续模板兔会增加更多实用的功能。 模板…

1.10 实战:Postman生成在线接口文档

对于接口测试,最最关键的一个部分,就是接口文档,有了详细的接口文档,我们才能根据接口文档去写好我们的接口测试用例以及写好我们的接口自动化测试。目前比较常用的接口文档是两种方式 Swagger我们看一下Swagger的官网的例子就可以看到它有多方便了。我们打开REST API Docu…

开源的数据流技术,该选择Redpanda还是Apache Kafka?

本文将比较Apache Kafka和Redpanda两种开源的数据流技术&#xff0c;在云原生实时处理能力上的不同&#xff0c;以及如何在项目中做出选择。 目前&#xff0c;Apache Kafka不但成为了数据流处理领域事实上的标准&#xff0c;而且带动了同类产品的出现。Redpanda就是其中之一…

2. 基础数据结构-数组

2. 基础数据结构-数组 2.1 概念 数组是一种数据结构&#xff0c;它是一个由相同类型元素组成的有序集合。在编程中&#xff0c;数组的定义是创建一个具有特定大小和类型的存储区域来存放多个值。数组可以是一维、二维或多维的。每个元素至少有一个索引或键来标识。 2.2 数组特…

C++——C++11(1)

时至今日&#xff0c;C标准已经到了C23&#xff0c;但是你要说哪一次提出的标准最经 典&#xff0c;那C11一定会被人提及&#xff0c;C11带来了数量可观的变化&#xff0c;其中包 含了约140个新特性&#xff0c;以及对C03标准中约600个缺陷的修正&#xff0c;这使得 C11更像是从…

no module named ‘xxx‘

目录结构如下 我想在GCNmodel的model里引入layers的GraphConvolution&#xff1a;from GCNmodel.layers import GraphConvolution&#xff0c;但这样却报错no module named GCNmodel&#xff0c;而且用from layers import GraphConvolution也不行。然后用sys.path.appen(xxx)…

echarts地图的常见用法:基本使用、区域颜色分级、水波动画、区域轮播、给地图添加背景图片和图标、3d地图、飞线图

前言 最近几天用echarts做中国地图&#xff0c;就把以前写的demo&#xff1a;在vue中实现中国地图 拿来用&#xff0c;结果到项目里直接报错了&#xff0c;后来发现是因为版本的问题&#xff0c;没办法只能从头进行踩坑了。以下内容基于vue3 和 echarts 5.32 基本使用 获取地…

广受好评的开源基础大模型最全梳理,你最钟意哪一个?

2023 年即将过去。一年以来&#xff0c;各式各样的大模型争相发布。当 OpenAI 和谷歌等科技巨头正在角逐时&#xff0c;另一方「势力」悄然崛起 —— 开源。 开源模型受到的质疑一向不少。它们是否能像专有模型一样优秀&#xff1f;是否能够媲美专有模型的性能&#xff1f; 迄…

go原生http开发简易blog(一)项目简介与搭建

文章目录 一、项目简介二、项目搭建前置知识三、首页- - -前端文件与后端结构体定义四、配置文件加载五、构造假数据- - -显示首页内容 代码地址&#xff1a;https://gitee.com/lymgoforIT/goblog 一、项目简介 使用Go原生http开发一个简易的博客系统&#xff0c;包含一下功能…

CCF-CSP真题《202309-5 阻击》思路+ c++满分题解

想查看其他题的真题及题解的同学可以前往查看&#xff1a;CCF-CSP真题附题解大全 试题编号&#xff1a;202309-5试题名称&#xff1a;阻击时间限制&#xff1a;2.0s内存限制&#xff1a;512.0MB问题描述&#xff1a; 问题描述 上回提到&#xff0c;西西艾弗岛下方有一个庞大的遗…

PyTorch自动梯度计算(注意点)

if params.grad is not None: params.grad.zero_() 我们实际的运算往往会涉及到若干个requires-grad为true的张量进行运算&#xff0c;在这种情况下&#xff0c;Pytorch会计算整个计算图上的损失的导数&#xff0c;并把这些结果累加到grad属性中。多次调用backward()会导致梯度…

chrome升级后,调试vue在控制台输出总是显示cjs.js

当前chrome版本120.0.6099.72 在vue中使用console.log输出时&#xff0c;总是显示cjs.js多少多少行&#xff0c;不能显示源文件名及行数 【解决方案】 打开控制台的设置 左侧找到“Ignore List”&#xff0c;取消勾选"enable Lgnore Listing"&#xff0c;并重启chr…

基于C/C++的非系统库自定义读写ini配置

INI文件由节、键、值组成。 节 [section] 参数 &#xff08;键值&#xff09; namevalue 这里将常用的操作方式封装成了一个dll供外部使用 // 下列 ifdef 块是创建使从 DLL 导出更简单的 // 宏的标准方法。此 DLL 中的所有文件都是用命令行上定义的 LIBCFG_EXPORTS // 符号…