题目链接

https://www.nowcoder.com/practice/e8a1b01a2df14cb2b228b30ee6a92163?tpId=13&tqId=11181&tPage=2&rp=2&ru=/ta/coding-interviews&qru=/ta/coding-interviews/question-ranking

题目描述

数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。

解题思路

  • 思路一:

    排序再计数,时间复杂度O(nlogn)。
  • 思路二:

    思路:利用快排的partion函数。

    算法描述:记数组长度为n,数组第n/2大的数即中位数即是要找的数。利用快排思想,每一轮partition后,定位一个数(位置也表示第几小),将该数的位置与中位比较,若该数的位置大于中位,则应partion该数左侧的这些比该数小的数。反之partion右侧。直到partion的这个数位置是中位。

    时间复杂度O(n),但会改变输入
  • 思路三:

    思路:由题,所求数比其他所有数出现次数还要多。

    算法描述:遍历数组,记两个变量,一个是临时元素变量,一个是次数,当便利到的的元素与临时元素变量相同则次数++,相异则次数--;若次数已为0,则临时元素变量换为下一个元素,次数设为1;这样最终的临时元素所存元素即为所求值。

    时间复杂度O(n)

关于特例:若输入数组中没有出现次数超过一半的数,则返回0.对于思路二的解法,这个要在找到第n/2大数后判断一下。

相关知识

我认为,关于快排,精髓在于partion函数:即每次定位一个数,即是最终排序的位置,且该数左侧的数都比它小,右侧的数都比它大。

思路二代码 O(n)

class Solution {
public:
int MoreThanHalfNum_Solution(vector<int> numbers) {
if (numbers.empty()) {
return 0;
}
int index = partition(numbers, 0, numbers.size() - 1);
int mid = numbers.size() >> 1;
while (index != mid) {
if (index > mid) {
index = partition(numbers, 0, index - 1);
}
else if (index < mid) {
index = partition(numbers, index + 1, numbers.size());
}
} if (!moreThanHalfCheck(numbers, numbers[index])) {
return 0;
}
else {
return numbers[index];
} }
private:
int partition(vector<int>& num, int start, int end) {
if (num.empty() || start > end || start < 0) {
throw "invaild!";
}
int index = randomInRange(start, end);
swap(num[start], num[index]); int l = start;
int r = end;
int tempElm = num[l];
while (l<r)
{
while (num[r] >= tempElm && l < r) {
--r;
}
if (l != r) {
num[l] = num[r];
++l;
}
while (num[l] <= tempElm && l < r) {
++l;
}
if (l != r) {
num[r] = num[l];
--r;
}
}
num[l] = tempElm;
return l;
} int randomInRange(int start, int end) {
if (start > end) {
throw"invaild!";
}
srand((unsigned)time(0));
int index = start + rand() % (end - start + 1);
return index;
} void swap(int& a, int& b) {
int temp = a;
a = b;
b = temp;
} bool moreThanHalfCheck(vector<int> numbers, int number) {
int numberCnt = 0;
for (int i = 0;i < numbers.size();++i) {
if (numbers[i] == number) {
++numberCnt;
}
}
if (numberCnt > numbers.size() / 2) {
return true;
}
else {
return false;
}
}
};

思路三代码

class Solution {
public:
int MoreThanHalfNum_Solution(vector<int> numbers) {
if (numbers.empty()) {
return 0;
}
int num;
int numCnt = 0;
for (int i = 0;i < numbers.size();++i) {
if (numCnt == 0) {
num = numbers[i];
++numCnt;
}
else {
if (numbers[i] == num) {
++numCnt;
}
else {
--numCnt;
}
}
}
if (!moreThanHalfCheck(numbers,num)) {
return 0;
}
else {
return num;
}
}
private:
bool moreThanHalfCheck(vector<int> numbers, int number) {
int numberCnt = 0;
for (int i = 0;i < numbers.size();++i) {
if (numbers[i] == number) {
++numberCnt;
}
}
if (numberCnt > numbers.size() / 2) {
return true;
}
else {
return false;
}
}
};

[剑指Offer]39-数组中出现次数超过一半的数字(快排延申,找第k大数同理)的更多相关文章

  1. 剑指 Offer 39. 数组中出现次数超过一半的数字 + 摩尔投票法

    剑指 Offer 39. 数组中出现次数超过一半的数字 Offer_39 题目描述 方法一:使用map存储数字出现的次数 public class Offer_39 { public int majo ...

  2. 剑指 Offer 39. 数组中出现次数超过一半的数字

    剑指 Offer 39. 数组中出现次数超过一半的数字 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 你可以假设数组是非空的,并且给定的数组总是存在多数元素. 示例 1: 输入: [ ...

  3. 力扣 - 剑指 Offer 39. 数组中出现次数超过一半的数字

    题目 剑指 Offer 39. 数组中出现次数超过一半的数字 思路1(排序) 因为题目说一定会存在超过数组长度一半的一个数字,所以我们将数组排序后,位于length/2位置的一定是众数 代码 clas ...

  4. 【Java】 剑指offer(39) 数组中出现次数超过一半的数字

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如 ...

  5. 每日一题 - 剑指 Offer 39. 数组中出现次数超过一半的数字

    题目信息 时间: 2019-06-29 题目链接:Leetcode tag: 数组 哈希表 难易程度:简单 题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 假设数组是非空的 ...

  6. 剑指Offer:数组中出现次数超过一半的数字【39】

    剑指Offer:数组中出现次数超过一半的数字[39] 题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如,输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于这 ...

  7. 【剑指Offer】数组中出现次数超过一半的数字 解题报告(Python)

    [剑指Offer]数组中出现次数超过一半的数字 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-inter ...

  8. Go语言实现:【剑指offer】数组中出现次数超过一半的数字

    该题目来源于牛客网<剑指offer>专题. 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组 ...

  9. 剑指OFFER之数组中出现次数超过一半的数字(九度OJ1370)

    题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2 ...

  10. 剑指Offer 28. 数组中出现次数超过一半的数字 (数组)

    题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...

随机推荐

  1. 深度学习原理与框架-卷积网络细节-经典网络架构 1.AlexNet 2.VGG

    1.AlexNet是2012年最早的第一代神经网络,整个神经网络的构架是8层的网络结构.网络刚开始使用11*11获得较大的感受野,随后使用5*5和3*3做特征的提取,最后使用3个全连接层做得分值得运算 ...

  2. ABAP-异常捕获

    异常处理 基于类的异常exception classes 捕获 使用老式方式捕获catchable runtime errors 向上抛出异常 手动触发异常(类异常) 6.1版本以后,TRY…ENDT ...

  3. 配置maven访问nexus,配置项目pom.xml以发布maven项目到nexus中

    maven访问nexus有三种配置方法,分别为: 项目pom.xml,优先级最高: user的settings.xml,优先级中,未在pom.xml中配置repository标签,则使用这个配置: m ...

  4. Servlet基本_Filter

    1.概念・サーブレットフィルタとは.サーブレットやJSPの「共通の前後処理」を記述するための仕組みです.・フィルタはサーブレットやJSPの前に位置し.実行前と後に.リクエスト.レスポンスに対して任意の ...

  5. Delphi中TApplication详解(转仅供自己参考)

    转自:http://blog.sina.com.cn/s/blog_4d6f55d90100bmv9.html TApplication是用于Delphi应用程序的类型,该类在单元forms中声明.T ...

  6. img标签在div里上下居中

    方法一:图片尺寸未知,IE8-不支持 CSS部分: <style> .content{ width:500px; height:500px; border:1px solid black; ...

  7. bug提单规范

    一.提单模板 标题:[项目组][模块][子模块][发生原因]问题简要描述描述:[预置条件] 有就写清楚,没有就写无[操作步骤]1.XXXXX2.XXXXXX3.XXXXX[实际结果] XXXXX[预期 ...

  8. 二叉堆复习(包括d堆)

    要期中考了……我真的是什么也不会啊,书都没看过TAT. 好吧整理一下二叉堆,这里就以最大堆为例好了. 首先二叉堆其实是一棵CBT,满足父节点的键值大于左右子节点的键值(wikipedia把这个叫键值, ...

  9. 如何遍历Set对象

    对 set 的遍历 1.迭代遍历: Set<String> set = new HashSet<String>(); Iterator<String> it = s ...

  10. Structs复习 访问web元素

    Structs帮我们在action和http里建立了联系 主要有四种方式 我们主要用第二种(IOC 依赖容器注入 ) Jar包 web.XML <?xml version="1.0&q ...