首页 > 编程语言 >Go 1.19 排序算法

Go 1.19 排序算法

时间:2023-09-30 14:31:49浏览次数:46  
标签:插入排序 元素 1.19 序列 pdqsort Go 排序 快速

插入排序(InsertionSort)

插入排序是一种简单直观的排序算法,它的基本思想是将待排序的元素插入到已经排好序的序列中,从而得到一个新的有序序列。插入排序的具体过程如下:

  1. 从第一个元素开始,认为它已经是有序的序列。
  2. 取出下一个元素,在已经排序的序列中从后向前扫描。
  3. 如果已经排序的序列中的元素大于新元素,将该元素移到下一个位置。
  4. 重复步骤3,直到已经排序的序列中的元素小于等于新元素。
  5. 将新元素插入到该位置后。
  6. 重复步骤2~5,直到所有元素都排序完成。

时间复杂度为O(n^2),空间复杂度为O(1),对于小规模的数据集来说,插入排序的效率是比较高的。

快速排序(QuickSort)

快速排序是一种基于分治思想的排序算法,它的基本思想是将待排序的序列分成两个子序列,其中一个子序列的所有元素都比另一个子序列的元素小,然后对这两个子序列分别进行排序,最终将它们合并成一个有序序列。快速排序的具体过程如下:

  1. 选择一个基准元素,通常是待排序序列的第一个元素。
  2. 将待排序序列分成两个子序列,其中一个子序列的所有元素都比基准元素小,另一个子序列的所有元素都比基准元素大。
  3. 对两个子序列分别进行快速排序,直到子序列中只剩下一个元素或为空。
  4. 将两个子序列合并成一个有序序列,其中基准元素放在两个子序列的中间位置。

时间复杂度为O(nlogn),最坏时间复杂度为O(n^2)

快速排序的效率比较高,因为它采用了分治的思想,可以将大规模的数据集分成小规模的数据集进行处理。

为了避免快速排序的最坏时间复杂度,可以采用随机化快速排序或者三路快排等算法来进行优化。

堆排序(HeapSort)

堆排序是一种基于堆的数据结构的排序算法,它的基本思想是将待排序的序列构建成一个堆,然后依次将堆顶元素取出来放入已排序序列中,最终得到一个有序序列。堆排序的具体过程如下:

  1. 将待排序的序列构建成一个堆,通常采用的是大根堆或小根堆。
  2. 将堆顶元素取出来,放入已排序序列中。
  3. 将堆的最后一个元素移动到堆顶,然后重新调整堆,使其满足堆的性质。
  4. 重复步骤2~3,直到堆中的元素全部取出来。

时间复杂度为O(nlogn),空间复杂度为O(n)

堆排序的效率比较高,因为它采用了堆的数据结构,可以快速的找到堆中的最大或最小元素。

堆排序是一种不稳定的排序算法,因为在构建堆的过程中可能会改变相同元素的相对位置。

对比

随机的情况下对比:

image.png

序列本身有序的情况下对比:

image.png

结论

  • 插入排序在短序列和序列有序的情况下最快
  • 大部分情况下,快速排序由较好的综合性能
  • 堆排序在任何情况下表现都比较好

pdqsort —— pattern-defeating-quicksort

pdqsort是一种快速、原地、稳定的排序算法,它是由Orson Peters于2019年提出的。pdqsort的原理是基于经典的快速排序算法,但它采用了一些新的技术来提高性能和稳定性。

pdqsort的主要思想是将快速排序分为两个阶段:

  1. 快速排序
  2. 插入排序
  • 在快速排序阶段,pdqsort使用经典的快速排序算法,选择一个中间元素作为枢轴(pivot),将数据分为两个子序列,并递归地对这两个子序列进行排序。但是,pdqsort在选择枢轴时采用了一些新的技术,如三点中值法(median-of-three),以避免最坏情况的发生。

  • 在插入排序阶段,pdqsort使用插入排序算法对小的子序列进行排序。插入排序是一种简单而有效的排序算法,它对小的子序列的排序效果很好。pdqsort通过在快速排序阶段和插入排序阶段之间进行平滑的转换,来保持排序的稳定性。

pdqsort还采用了一些其他的技术来提高性能和稳定性,如分区排序(partition sort)和双轴快速排序(dual-pivot quicksort)。这些技术使得pdqsort在处理大量数据时具有很好的性能,并且可以保持排序的稳定性。

标签:插入排序,元素,1.19,序列,pdqsort,Go,排序,快速
From: https://blog.51cto.com/u_12482515/7663897

相关文章

  • Go - closure
     packagemainimport"fmt"funcmain(){fori:=0;i<3;i++{fmt.Println(outerFunc()())}fmt.Println("------------------------")next:=outerFunc()fori:=0;i<3;i++{fmt.Pri......
  • Go - Logging to the System Log Service
    Problem: Youwanttologintothesystemloginsteadofyourlogfiles.Solution: Usethelog/syslogpackagetowritetosyslog. Syslogisastandardnetwork-basedloggingprotocol.Ithaslongbeenthedefactostandardforloggingsystemeventsand......
  • Go每日一库之147:goldmark(Markdown转html)
    简介使用Markdown书写结构化的文档和评论已经相当流行了,Web服务需要将用户编写的Markdown文本转换为html以便浏览器渲染,还常常需要对Markdown语法进行自定义扩展以实现个性化的功能。本期要介绍的**goldmark**就是Go生态中的一款Markdown解析器和扩展器,与GitHub......
  • 进化算法中的遗传算法(Genetic Algorithms)
    进化算法中的遗传算法(GeneticAlgorithms)引言进化算法是一类基于自然进化原理的优化算法,通过模拟生物进化过程中的选择、交叉和变异等操作,来求解复杂问题。遗传算法(GeneticAlgorithms)是进化算法中最为经典和常用的一种方法。本文将介绍遗传算法的基本原理、核心操作和应用领域,以及......
  • Go每日一库之169:dongle(编解码、加解密)
    一个轻量级、语义化、对开发者友好的golang编码解码、加密解密库。安装使用//使用github库goget-ugithub.com/golang-module/dongleimport("github.com/golang-module/dongle")//使用gitee库goget-ugitee.com/golang-module/dongleimport("g......
  • Go每日一库之168:redsync(redis分布式锁)
    今天给大家推荐的是基于redis的Go版本的分布式锁工具:redsync。该工具也是redis官网上推荐的。redsync基于redis的高可用、高性能、防死锁、防误删的分布式锁实现,具有高性能、高可用、防死锁、防误删的特点。一、分布式锁基础知识什么是分布式锁锁,在编程语言中就是一个变量,该变......
  • Go每日一库之167:emoji(emoji表情)
    大家在使用微信或钉钉聊天时,一定使用过表情符号。今天就给大家介绍一个能够在终端上显示emoji表情符号的包:emoji。实现原理:emoji表情符号实际上就是在unicode编码表中有定义的一个编码。通过将符号的文字表示和对应的unicode编码进行一一对应,在使用时对文字符号进行替换成rune字......
  • Go每日一库之187:singleflight(合并重复调用)
    本文主要介绍Go语言中的singleflight包,包括什么是singleflight以及如何使用singleflight合并请求解决缓存击穿问题。singleflight目前(Go1.20)还属于Go的准标准库,它提供了重复函数调用抑制机制,使用它可以避免同时进行相同的函数调用。第一个调用未完成时后续的重复调用会等待,当第......
  • Go每日一库之186:sonic(高性能JSON库)
    介绍我们在日常开发中,常常会对JSON进行序列化和反序列化。Golang提供了encoding/json包对JSON进行Marshal/Unmarshal操作。但是在大规模数据场景下,该包的性能和开销确实会有点不够看。在生产环境下,JSON序列化和反序列化会被频繁的使用到。在测试中,CPU使用率接近10%,其中极端情况......
  • Go每日一库之184:katana(新一代爬虫框架)
    项目链接https://github.com/projectdiscovery/katana项目简介katana是一个使用golang编写的新一代爬虫框架,支持HTTP和headless抓取网页信息不仅可以作为库集成到Golang项目,还可以通过命令行直接抓取,对于有一些轻量级的抓取任务的开发者配合jq一起使用简直就是福......