力扣-排序算法

news2024/9/27 21:30:54

排序算法,一般都可以使用std::sort()来快速排序。

这里介绍一些相关的算法,巩固记忆。

快速排序

跟二分查找有一丢丢像。

首先选择一个基准元素,一般就直接选择第一个。然后两个指针,一个指向第一个,一个指向最后一个。第一个现在是空,就从最后一个开始,跟基准元素做判断,小于基准元素的就放到第一个位置,然后第一指针往后移。按这种顺序直到两个指针相遇,相遇的位置放入基准元素。然后从基准元素劈开两半,再按相同方法排序。

void quicksort(vector<int> nums,int l,int r){
    if(l+1>=r)
        return;
    int first=l,last=r-1;
    int key=nums[first];
    while(first<=last){
        while(first<last&&nums[last]>=key)
            last--;
        nums[first]=nums[last];
        while(first<last&&nums[first]<=key)
            first++;
        nums[last]=nums[first];
    }
    nums[first]=key;
    quicksort(nums,l,first);
    quicksort(nums,first+1,r);
}

归并算法

归并算法是一种分治算法,先分再治。比如说8个数字。先分成4个4个,然后4个内部再继续分,分成2个2个。2个排好序后合并,4个排好序后再合并。

void merge_sort(vector<int> &nums,int l,int r,vector<int>& temp){
    if(l+1>=r)
        return ;
    int m=(l+r)/2;
    merge_sort(nums,l,m,temp);
    merge_sort(nums,m+1,r,temp);
    int a=l,b=m,i=l;
    while(a<m||b<r){
        if(nums[a]<nums[b])
            nums[i++]=nums[a++];
        else
            nums[i++]=nums[b++];
    }
    for(i=l;i<r;i++)
        nums[i]=temp[i];
}

插入排序

首先是一个数,本来就有序。接着两个数进行排序。接着三个数。

所谓插入,比如说八个数进行排序,其实前七个已经有序,这个时候只需要把第八个插入到前七个数的合适位置即可。怎么插入,就从后往前,一个一个比较,如果小于就往前移。

void insert_sort(vector<int>& nums,int n){
    int i,j;
    for(i=0;i<n;i++){
        for(j=i;j>0;j--){
            if(nums[j]>nums[j-1])
                swap(nums[j],nums[j-1]);
        }
    }
}

冒泡排序

相邻两个数比较,大的往后移,每一轮最大的数将会移到最后。

可以进行一个优化,可能出现一种情况,从第二轮过后,其实数组已经是有序的了,但是按照算法步骤来走的话,即使已经排好序了,但仍是会进行后边的比较,知道全部比较完成

因此,我们可以对代码进行优化,如果发现在某趟排序中,没有发生一次交换, 可以提前结束冒泡排序

解决方式:可以通过一个标志位来进行判断

void bubble_sort(vector<int>& nums,int n){
    int flag=false;
    int i,j;
    for(i=1;i<n;i++){
        flag=false;
        for(j=1;j<n-i+1;j++){
            if(nums[j]<nums[j-1]){
                swap(nums[j],nums[j-1]);
                flag=true;
            }     
        }
        if(flag==false)
            return ;
    }
}

选择排序

选择最小的数跟第一个数交换,按照同样的方法对后面的n-1个进行排序。

void select_sort(vector<int>& nums,int n){
    int i,j;
    for(i=0;i<n;i++){
        int m=i;
        for(j=i+1;j<n;j++){
            if(nums[j]<nums[m]){
                m=j;
            }
        }
        swap(nums[i],nums[m]);
    }
}

215.数组中的第K个元素

215. 数组中的第K个最大元素

给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。

请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。

你必须设计并实现时间复杂度为 O(n) 的算法解决此问题。

示例 1:

输入: [3,2,1,5,6,4], k = 2输出: 5

示例 2:

输入: [3,2,3,1,2,4,5,5,6], k = 4输出: 4

提示:

  • 1 <= k <= nums.length <= 10^{5}
  • -10^{4} <= nums[i] <= 10^{4}

题解

可以利用快速排序的思想来解决这个问题。

选择一个数,然后将大于它的数字往后移,小于的往前移。如果这个刚好是第k位,直接返回。如果不是,大于k,则对前面做相同的排序,如果不是就对后面做相似的排序。这里的k指的是第k大,因此从小到大排就是第n-k位。

class Solution {
public:
    int quickselect(vector<int> &nums,int l,int r,int k){
        if(l==r)
            return nums[k];
        int p=nums[l],i=l-1,j=r+1;
        while(i<j){
            do i++; while (nums[i] < p);
            do j--; while (nums[j] > p);
            if(i<j)
                swap(nums[i],nums[j]);
        }
        if(k<=j) 
            return quickselect(nums,l,j,k);
        else 
            return quickselect(nums,j+1,r,k);
    }
    int findKthLargest(vector<int> &nums, int k) {
        int n=nums.size();
        return quickselect(nums,0,n-1,n-k);
    }
};

347.前k个高频元素

347. 前 K 个高频元素

题目

给你一个整数数组 nums 和一个整数 k ,请你返回其中出现频率前 k 高的元素。你可以按 任意顺序 返回答案。

示例 1:

输入: nums = [1,1,1,2,2,3], k = 2
输出: [1,2]

示例 2:

输入: nums = [1], k = 1
输出: [1]

提示:

  • 1 <= nums.length <= 10^{5}
  • k 的取值范围是 [1, 数组中不相同的元素的个数]
  • 题目数据保证答案唯一,换句话说,数组中前 k 个高频元素的集合是唯一的

题解

可以使用桶排序,使用STL库中的unordered_map,提供了一种基于哈希表的键值对容器。

可以对每一个数字出现的次数进行计算并且存储。

    unordered_map<int,int> m;
        for(int num:nums) m[num]++;

接着进行排序,然后再定义一个新的数组用来存储要返回的数组。

class Solution {
public:
    vector<int> topKFrequent(vector<int>& nums, int k) {
        unordered_map<int,int> m;
        for(int num:nums) m[num]++;
        vector<pair<int,int>> s;
        for(auto t:m)
            s.push_back({t.second,t.first});
        sort(s.begin(),s.end());
        vector<int> ans;
        int t=s.size()-1;
        while(k--){
            ans.push_back(s[t--].second);
        }
        return ans;
    }
};

排序这里也有一个库可以使用。

<priority_queue> 是标准模板库(STL)的一部分,用于实现优先队列。

优先队列是一种特殊的队列,它允许我们快速访问队列中具有最高(或最低)优先级的元素。

class Solution {
public:
    vector<int> topKFrequent(vector<int>& nums, int k) {
        unordered_map<int,int> m;
        for(int num:nums) m[num]++;
        priority_queue<pair<int,int>> s;
        for(auto i:m) s.emplace(i.second,i.first);
        vector<int> ans;
        while(k--){
            ans.push_back(s.top().second);
            s.pop();
        }
        return ans;
    }
};

不得不说如果熟悉使用STL库,真的省事很多。

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

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

相关文章

绝区玖--人工智能物料清单 (AI BOM)

前言 AI BOM 涵盖了从输入模型的数据到为模型提供支持的基础设施以及将 AI 从概念转化为生产的过程的一切。 但为什么我们需要人工智能物料清单&#xff1f;答案在于当今世界人工智能/Gen AI系统的复杂性和关键性&#xff1a; 透明度和可重复性&#xff1a;AI BOM 提供所有组件…

MES系统在装备制造行业核心应用场景介绍

MES软件在企业中有着广泛的应用场景&#xff0c;主要包括生产计划排程、生产过程监控、质量管理、设备管理、库存管理、数据分析等领域。 通过实时监控生产过程、收集数据、进行分析&#xff0c;MES软件可以帮助企业实现生产过程可视化、透明化&#xff0c;提高生产效率&#…

开源项目有哪些机遇与挑战

目录 1.概述 2.开源项目的发展趋势 2.1. 开源项目的发展现状 2.2. 开源社区的活跃度 2.3. 开源项目在技术创新中的作用 3.参与开源的经验分享 3.1. 选择开源项目 3.2. 理解项目结构和文档 3.3. 贡献代码 3.4. 与开源社区的合作 3.5. 学习和成长 4.开源项目的挑战 …

CSS【详解】文本相关样式(含 font 系列样式,文本颜色 color,三种颜色表示法,文本排版-含最佳实战范例,文本装饰,分散对齐,渐变色文本等)

文本风格 font-style font-style:italic 值描述normal默认值。浏览器显示一个标准的字体样式。italic加载对应字体的斜体字体文件&#xff0c;若找不到斜体字体文件&#xff0c;则进行物理上的倾斜。 标签默认font-style:italicoblique浏览器会显示一个倾斜的字体样式。 文本粗…

华为HCIP Datacom H12-821 卷33

1.判断题 缺省情况下&#xff0c;华为AR路由器的VRRP运行在抢占模式下 A、对 B、错 正确答案&#xff1a; A 解析&#xff1a; 无 2.判断题 一个Route-Policy下可以有多个节点&#xff0c;不同的节点号用节点号标识&#xff0c;不同节点之间的关系是"或"的关…

什么是 C 语言中的宏定义?

&#x1f345;关注博主&#x1f397;️ 带你畅游技术世界&#xff0c;不错过每一次成长机会&#xff01; &#x1f4d9;C 语言百万年薪修炼课程 通俗易懂&#xff0c;深入浅出&#xff0c;匠心打磨&#xff0c;死磕细节&#xff0c;6年迭代&#xff0c;看过的人都说好。 文章目…

告别缓慢下载,Cloudflare带你体验极速Docker镜像加速

背景 国内的Docker镜像服务似乎突然进入了寒冬&#xff0c;不仅Docker镜像服务受到了影响&#xff0c;连NPM镜像也可能面临下架的风险。这对依赖这些服务的开发者们来说&#xff0c;无疑是一个不小的困扰。 近日&#xff0c;SJTUG&#xff08;上海交通大学Linux用户组&#x…

Apache Hadoop之历史服务器日志聚集配置

上篇介绍了Apache Hadoop的分布式集群环境搭建&#xff0c;并测试了MapReduce分布式计算案例。但集群历史做了哪些任务&#xff0c;任务执行日志等信息还需要配置历史服务器和日志聚集才能更好的查看。 配置历史服务器 在Yarn中运行的任务产生的日志数据不能查看&#xff0c;…

30. 01背包问题 二维,01背包问题 一维,416.分割等和子集

背包问题分类&#xff1a; 1、确定dp数组以及下标的含义对于背包问题&#xff0c;有一种写法&#xff0c; 是使用二维数组&#xff0c;即dp[i][j] 表示从下标为[0-i]的物品里任意取&#xff0c;放进容量为j的背包&#xff0c;价值总和最大是多少。2、确定递推公式&#xff0c;…

硅谷甄选二(登录)

一、登录路由静态组件 src\views\login\index.vue <template><div class"login_container"><!-- Layout 布局 --><el-row><el-col :span"12" :xs"0"></el-col><el-col :span"12" :xs"2…

Qt基础控件总结—多页面切换(QStackWidget类、QTabBar类和QTabWidget类)

QStackedWidget 类 QStackedWidget 类是在 QStackedLayout 之上构造的一个便利的部件,其使用方法与步骤和 QStackedLayout 是一样的。QStackedWidget 类的成员函数与 QStackedLayout 类也基本上是一致的,使用该类就和使用 QStackedLayout 一样。 使用该类可以参考QStackedL…

施罗德数列SQL实现

在组合数学中,施罗德数用来描述从(0,0)到(n,n)的格路中,只能使用(1,0)、(0,1)、(1,1)三种移动方式,始终位于对角线下方且不越过对角线的路径数 DECLARE n INT 10 DECLARE i INT DECLARE rst INT DECLARE old INT1CREATE TABLE #rst (i INT ,rst int )INSERT INTO #rst values(…

Python 中创建当前日期和时间的文件名技巧详解

概要 在日常开发中,经常需要创建带有当前日期和时间的文件名,以便进行日志记录、数据备份或版本控制等操作。Python 提供了丰富的库和函数,可以方便地获取当前日期和时间,并将其格式化为字符串,用于生成文件名。本文将详细介绍如何使用 Python 创建带有当前日期和时间的文…

springboot大学生竞赛管理系统-计算机毕业设计源码37276

摘 要 随着教育信息化的不断发展&#xff0c;大学生竞赛已成为高校教育的重要组成部分。传统的竞赛组织和管理方式存在着诸多问题&#xff0c;如信息不透明、效率低下、管理不便等。为了解决这些问题&#xff0c;提高竞赛组织和管理效率&#xff0c;本文设计并实现了一个基于Sp…

STM32(二):STM32工作原理

0、参考1、寄存器和存储器基本概念&#xff08;1&#xff09;基本概念&#xff08;2&#xff09;主要区别&#xff08;3&#xff09;联系&#xff08;4&#xff09;实际应用中的案例&#xff08;5&#xff09;总结&#xff08;6&#xff09;一些名词解释 2、STM32指南者板子-存…

免费GPU——Google Colab使用

免费GPU——Google Colab使用 1、创建新的Notebook 网址&#xff1a;https://colab.research.google.com/ 点击“新建笔记本”进行创建 2、设置免费GPU 点击“更改运行时类型”&#xff0c;打开界面如下所示&#xff1a; 选择“T4 GPU”&#xff0c;然后“保存”即可使用…

秒速将油管视频转换为博客文章!

摘要&#xff1a; 本文提供了一个免费试用的分步指南&#xff0c;介绍如何在短时间内将YouTube视频内容转换为博客文章&#xff0c;以扩展网络营销效果。通过使用特定的模板和自动化工具&#xff0c;可以显著提高内容转换的效率。 关键词&#xff1a; YouTube视频&#xff0c;…

会员运营体系设计及SOP梳理

一些做会员的经验和方法分享给大家&#xff0c;包括顶层思考、流程的梳理、组织的建立&#xff0c;后续会做成系列&#xff0c;最近几期主要围绕顶层策略方面&#xff0c;以下是核心内容的整理&#xff1a; 1、会员运营体系设计 顶层设计与关键业务定位&#xff1a;建立客户运营…

中霖教育:考完一建后二建证书还有用吗?

通过一级建造师考试后&#xff0c;二建证书依然有效。如果一建和二建证书是不同专业&#xff0c;通过一级建造师考试&#xff0c;二级建造师资格同样保持有效。对于相同专业的情况&#xff0c;两种证书亦也在相同单位同时注册&#xff0c;不会注销。 一级与二级建造师的区别&a…

python入门综合篇—资源爬取与exe打包(图形界面)

了解我的人都知道&#xff0c;我是一个谨言慎行且兴趣爱好广泛的IT&#xff0c;作为一个合格的前端&#xff0c;没事捣鼓一下python很合理吧&#xff0c;再没事搞搞java和php也很合乎逻辑吧&#xff0c;实在没事&#xff0c;玩玩linux服务器也是合乎常理的吧。所以&#xff0c;…