力扣-414.第三大的数(两种解法)

2023-11-22 20:01
文章标签 力扣 两种 解法 第三 414

本文主要是介绍力扣-414.第三大的数(两种解法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 第三大的数
      • 解法一(排序加遍历对比)
      • 解法二(遍历一遍加迭代)


第三大的数

题目:

给你一个非空数组,返回此数组中第三大的数 。如果不存在,则返回数组中最大的数。

示例 1: 输入:[3, 2, 1] 输出:1
解释:第三大的数是 1 。

示例 2: 输入:[1, 2] 输出:2
解释:第三大的数不存在, 所以返回最大的数 2 。

示例 3: 输入:[2, 2, 3, 1] 输出:1
解释:注意,要求返回第三大的数,是指在所有不同数字中排第三大的数。 此例中存在两个值为
2 的数,它们都排第二。在所有不同数字中排第三大的数为 1 。

解法一(排序加遍历对比)

分析:

1.因为是要第三大的数,我们可以先判断数组的长度,如果为 1 则直接放回数组的第一个数,为2就判断哪个大在返回最大值。
2.当我们判断数组长度大于等于3后可以将它们进行降序排序,然后再用循环判断是否有三个数不相等(因为有可能整个数组都是同一个数或者是由两个数组成),然后再输出第三大的元素

在写代码之前我们先来简单介绍一个排序的库函数 qsort

对所指向的数组元素进行排序,每个元素的长度为字节,使用函数确定顺序。
此函数使用的排序算法通过调用指定的函数来比较元素对,并将指向元素的指针作为参数。
该函数不返回任何值,而是通过对数组的元素进行重新排序来修改所指向的数组的内容.

在这里插入图片描述
注意的是 compar函数的创建的格式和 qsort的一样: int (*compar)(const void*,const void*));//这个是自己创建的
在这里插入图片描述

qsort的使用

void qsort (void* base, size_t num, size_t size,   int (*compar)(const void*,const void*));

我们来实现一个整形排序

int hanshu(const void* p1,const void* p2) {//这里要按照qsort中的函数参数类型来写 
//int (*compar)(const void*,const void*));return  (*((int*)p1) - *((int*)p2));//返回-上面有解释  (int*)p1--强制转换为int类型指针
}
int main() {int arr[] = { 5,3,2,4,1 };int zs = sizeof(arr) / sizeof(arr[0]);//数组大小qsort(arr, zs, sizeof(arr[0]), hanshu);//按照规则填入, sizeof(arr[0])一个元素占多少字节for (int i = 0; i < zs; i++)//打印printf("%d ", arr[i]);return 0;
}

我们来看看运行结果:
在这里插入图片描述
成功完成排序
好,我们接下来就通过代码实现:

int te(const void* p1, const void* p2) {//排序判断大小的函数return *(( int*)p2) >*(( int*)p1);//返回 > 就交换
}int thirdMax(int* nums, int numsSize) {if (numsSize == 1) {//判断长度return nums[0];}else if (numsSize == 2) {//判断长度,只有2个元素,判断大小再返回int max = nums[0];if (max < nums[1])max = nums[1];return max;}else {qsort(nums, numsSize, sizeof(int), te);//实现排序,一定要按照函数的格式放值int r = 0,t;//r是用来判断是否有三个不同的数,t是保留第三大数的下标的for (int j = 0; j < numsSize-1; j++) {if (nums[j] != nums[j + 1]) {//前后不一样r++r++;if (r == 2) {//当r==2时证明有三个不同的元素了,此时可以结束判断了t = j+1;//记录下标break;}}
}if (r != 2)//判断是否够三个数,不够直接返回最大值return nums[0];else//r==2返回第三大的元素return nums[t];}}

以上就是第一种解法的详细代码和注释了

解法二(遍历一遍加迭代)

分析:

1.我们只要求前三大的元素,那么我们可以设置三个变量分别代表最大,第二,第三大的元素。
2.再对所有元素判断,判断在我们设置的三个变量的哪个区间里,再进行迭代
3.最后再判断第三大的变量是否被改变,要是被改变了就是返回改值,不然返回最大值

我们先介绍一下
int、long、longlong等的最大值和最小值的宏定义(部分)
头文件:<limits.h>

#define SHRT MIN	(-32768)//short-min
#define SHRT_MAX	32767	//short-max
#define USHRT_MAX	0xffff	//无符号 short-max
#define INT MIN	(-2147483647-1)//int-min	
#define INT_MAX	2147483647	//int-max
#define UINT_MAX	0xffffffff	//无符号 int-max
#define LONG MIN	(-2147483647L -1)//long-min	
#define LONG MAX	2147483647L	//long-max
#define ULONG_MAX	0xffffffffUL// 无符号long-man	
#define LLONG MAX	9223372036854775807i64	//long long-man
#define LLONG MIN	(-9223372036854775807i64 -1)//long long-min
#define ULLONG MAX	0xffffffffffffffffui64//无符号 longlong-max

代码:

int thirdMax(int* nums, int numsSize) {long a = LONG_MIN;//最大元素---这里用long是因为力扣那个案例里有大于int类型的值,LONG_MIN--long的最小值long b= LONG_MIN;//第二个大元素long c = LONG_MIN;//第三大元素for (int i = 0; i < numsSize; i++) {if (nums[i] > a) {//比最大值都大,将数值由大往小交换c = b;b = a;a = nums[i];}else if (nums[i]<a && nums[i]>b) {//在最大值和第二大的中间,这时不动最大值,其他往下交换c = b;b = nums[i];}else if (nums[i]<b && nums[i]>c) {//在第二值和第三大的中间,这时不动最大值和第二大,将第三换掉c = nums[i];}}return c == LONG_MIN ? a : c;//判断第三个值是否是原值
}

以上就是第二种解法的详细代码和注释了

今天的分享就到这,谢谢大家观看!

这篇关于力扣-414.第三大的数(两种解法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/412574

相关文章

Python读取TIF文件的两种方法实现

《Python读取TIF文件的两种方法实现》本文主要介绍了Python读取TIF文件的两种方法实现,包括使用tifffile库和Pillow库逐帧读取TIFF文件,具有一定的参考价值,感兴趣的可以了解... 目录方法 1:使用 tifffile 逐帧读取安装 tifffile:逐帧读取代码:方法 2:使用

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

两数之和--力扣1

两数之和 题目思路C++代码 题目 思路 根据题目要求,元素不能重复且不需要排序,我们这里使用哈希表unordered_map。注意题目说了只对应一种答案。 所以我们在循环中,使用目标值减去当前循环的nums[i],得到差值,如果我们在map中能够找到这个差值,就说明存在两个整数的和为目标值。 如果没有找到,就将当前循环的nums[i]以及下标i放入map中,以便后续查

LeetCode 第414场周赛个人题解

目录 Q1. 将日期转换为二进制表示 原题链接 思路分析 AC代码 Q2. 范围内整数的最大得分 原题链接 思路分析 AC代码 Q3. 到达数组末尾的最大得分 原题链接 思路分析 AC代码 Q4. 吃掉所有兵需要的最多移动次数 原题链接 思路分析 AC代码 Q1. 将日期转换为二进制表示 原题链接 Q1. 将日期转换为二进制表示 思路分析

力扣第347题 前K个高频元素

前言 记录一下刷题历程 力扣第347题 前K个高频元素 前K个高频元素 原题目: 分析 我们首先使用哈希表来统计数字出现的频率,然后我们使用一个桶排序。我们首先定义一个长度为n+1的数组,对于下图这个示例就是长度为7的数组。为什么需要一个长度为n+1的数组呢?假如说总共有三个数字都为1,那么我们需要把这个1放在数组下标为3的位置,假如说数组长度为n,对于这个例子就是长度为3,那么它的

android两种日志获取log4j

android   log4j 加载日志使用方法; 先上图: 有两种方式: 1:直接使用架包 加载(两个都要使用); 架包:android-logging-log4j-1.0.3.jar 、log4j-1.2.15.jar  (说明:也可以使用架包:log4j-1.2.17.jar)  2:对架包输入日志的二次封装使用; 1:直接使用 log4j 日志框架获取日志信息: A:配置 日志 文

c++11工厂子类实现自注册的两种方法

文章目录 一、产品类构建1. 猫基类与各品种猫子类2.狗基类与各品种狗子类 二、工厂类构建三、客户端使用switch-case实现调用不同工厂子类四、自注册方法一:公开注册函数显式注册五、自注册方法二:构造函数隐形注册总结 一、产品类构建 1. 猫基类与各品种猫子类 class Cat {public:virtual void Printer() = 0;};class

【数据结构与算法 | 灵神题单 | 删除链表篇】力扣3217, 82, 237

总结,删除链表节点问题使用到列表,哈希表,递归比较容易超时,我觉得使用计数排序比较稳,处理起来也不是很难。 1. 力扣3217:从链表中移除在数组中的节点 1.1 题目: 给你一个整数数组 nums 和一个链表的头节点 head。从链表中移除所有存在于 nums 中的节点后,返回修改后的链表的头节点。 示例 1: 输入: nums = [1,2,3], head = [1,2,3,

力扣 739. 每日温度【经典单调栈题目】

1. 题目 理解题意: 1.1. 给一个温度集合, 要返回一个对应长度的结果集合, 这个结果集合里面的元素 i 是 当前 i 位置的元素的下一个更高温度的元素的位置和当前 i 位置的距离之差, 若是当前元素不存在下一个更高温度的元素, 则这个位置用0代替; 2. 思路 本题用单调栈来求解;单调栈就适用于来求当前元素左边或者右边第一个比当前元素大或者小的元素;【单调栈:让栈中的元素保持单调