剑指Offer——数组中出现次数超过一半的数字——一题多解
看题目:
数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。
我的直接思路:
用map计数,简单直接,遍历一次数组,用hashmap记录,key为int值,value为出现次数;
第二次再用map.entrySet找出有没value大于数组长度一般的entry,有的话返回它的key。
时间复杂度也是2n而已,这个方法时间复杂度是O(n)空间复杂度也是O(n)
代码实现:
/*方法1
蛮力,遍历一次,用一个map来记录
第二次遍历把出现次数大于length/2的那个值找出来
*/ public int MoreThanHalfNum_Solution(int [] array) {
int targetCount = array.length / 2; //key为数组中的值,value为出现的次数
HashMap<Integer,Integer> map = new HashMap<Integer,Integer>();
for(int temp : array) {
if(map.get(temp) == null)map.put(temp, 1);
else map.put(temp, map.get(temp) + 1);
}
Set<Entry<Integer,Integer>> entrySet = map.entrySet();
for(Entry<Integer,Integer> entry : entrySet) {
if(entry.getValue() > targetCount)return entry.getKey();
}
return 0;
}
方法2:中位数
/*方法2
排序后中位数法
如果有个数字出现的次数大于数组长度的一半,那么这个数组排序后,它的中位数一定是这个数字
*/ public int MoreThanHalfNum_Solution(int [] array) {
Arrays.sort(array);
int count = 0, middleNum = array[array.length / 2];
for(int temp : array) {
if(temp == middleNum)count++;
}
if(count > array.length / 2)return middleNum;
else return 0;
}
方法3:——快排思路
方法2中,我们排序是为了找出中位数,那么如果可以更快地找出中位数就不用排序了。
通过Partition()返回index,如果index==mid,那么就表明找到了数组的中位数;如果index<mid,表明中位数在[index+1,end]之间;如果index>mid,表明中位数在[start,index-1]之间。直到最后求得index==mid循环结束。
/*方法3
利用快排的思想
*/ public int MoreThanHalfNum_Solution(int [] array) {
if(array.length <= 0)return 0; int begin = 0, end = array.length - 1, middle = array.length / 2;
int partition = partition(array, begin, end); while(partition != middle) {
if(partition > middle) {//中位数在partition的左边
partition = partition(array, begin, partition - 1);
} else {//中位数在partition右边
partition = partition(array, partition + 1, end);
}
} //找出中位数了,看这个中位数出现的次数是否符合要求
int count = 0, middleKey = array[middle];
for(int temp : array) {
if(temp == middleKey)count++;
} if(count > array.length / 2)return array[middle];
else return 0;
} //这个方法是以第一个元素为基准,然后进行划分,划分后比基准元素小的数字都在它左边,比它大的数字都在它右边
返回划分后,这个元素的新index//
private int partition(int[] a, int begin, int end) {
int key = a[begin]; int i = begin, j = end;
while(i < j) {
while(i < j && a[j] >= key)j--;
while(i < j && a[i] <= key)i++;
swap(a, i, j);
}
swap(a, begin, i);
return i;
} //交换数字函数,传入数组还有要交换的两个数字的index//
private void swap(int[] a, int first, int second) {
int temp = a[first];
a[first] = a[second];
a[second] = temp;
}
遇到相同元素,count++;
遇到不相同元素,即为敌人,同归于尽,count--;当遇到count为0的情况,又以新的i值作为守阵地的士兵,继续下去,同时count更新为1.
当我们遍历到下一个数字的时候,
如果下一个数字和当前我们保存的数字相同,则次数加 1;
如果和当前我们保存的数字不同,则次数减 1;
当次数减到 0 的时候,我们将保存的数字改为当前遍历所处的位置,并将次数更改为 1。
看代码:
/*方法4
阵地攻守思想,其实和牛客上面那个什么“用户分形叶”的思路一样的,不同实现而已
*/
public int MoreThanHalfNum_Solution(int [] array) {
//int count = 1;
int count = 0;//这里我们设初始的count为0好了,因为利用java的foreach语法是要从第一个key开始遍历的,那么就是碰到第一个开始,设为1
int key = array[0];
for(int temp : array) {
if(temp == key)count++;
else if(count > 0)count--;
else {//count==0的情况,这个时候把key换成现在这个元素,并把count设为1,意思是这是第一次碰到这个元素
key = temp;
count = 1;
}
} //判断这个得到的key是不是符合要求
int count2 = 0;
for(int temp : array) {
if(temp == key)count2++;
}
if(count2 > array.length / 2)return key;
else return 0;
}
剑指Offer——数组中出现次数超过一半的数字——一题多解的更多相关文章
- python剑指offer数组中出现次数超过一半的数字
题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...
- 剑指offer——数组中出现次数超过一半的数字(c++)
题目描述数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如 ...
- 剑指Offer——数组中出现次数超过一半的数字
题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2 ...
- 用js刷剑指offer(数组中出现次数超过一半的数字)
题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...
- 剑指Offer-28.数组中出现次数超过一半的数字(C++/Java)
题目: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如 ...
- 剑指offer--11.数组中出现次数超过一半的数字
unique(), count()函数好用 ---------------------------------------------------------------------- 时间限制:1秒 ...
- 1-剑指offer: 数组中出现次数超过一半的数字
题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...
- 剑指offer-数组中出现次数超过一半的数字
题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...
- Leetcode - 剑指offer 面试题29:数组中出现次数超过一半的数字及其变形(腾讯2015秋招 编程题4)
剑指offer 面试题29:数组中出现次数超过一半的数字 提交网址: http://www.nowcoder.com/practice/e8a1b01a2df14cb2b228b30ee6a92163 ...
随机推荐
- 【转】 Pro Android学习笔记(六三):Preferences(7):代码控制首选项
[-] 代码实现preference 利用preference保存状态 DialogPreference 代码实现preference View可以不通过xml进行设置,有代码直接进行设置,首选项pr ...
- ETL之Tungsten Replicator
1 概述 1.1 介绍 Tungsten Replicator是数据库集群和复制供应商Continuent推出的高性能.开源的数据复制引擎,是Continuent最先进的集群解决方案的核心组件之一,特 ...
- JavaScript-Tool:jquery.cookie.js
ylbtech-JavaScript-Tool:jquery.cookie.js 1.返回顶部 1.jquery.cookie.js /*! * jQuery Cookie Plugin v1.4.0 ...
- 关于使用C# 启动msi失败的问题
原以为在启动msi是件小儿科的事,上代码: ProcessStartInfo psi = new ProcessStartInfo(); psi.FileName = "C:\\myTest ...
- assert.ok()
测试 value 是否为真值. 相当于 assert.equal(!!value, true, message). 如果 value 不为真值,则抛出一个带有 message 属性的 Assertio ...
- hibernate框架内容整理 学习
1.1 ORM概述 Object Relation Mapping 对象关系映射. 对象-关系映射(OBJECT/RELATIONALMAPPING,简称ORM),是随着面向对象的软件开发方法发展而产 ...
- JavaScript学习系列2一JavaScript中的变量作用域
在写这篇文章之前,再次提醒一下 JavaScript 是大小写敏感的语言 // 'test', 'Test', 'TeSt' , 'TEST' 是4个不同的变量名 JavaScript中的变量,最重要 ...
- 【关于java多线程和socket通信的一些记录】---高并发/高负载/高可用/重入锁
多线程:提高cpu的使用效率,多线程是指在同一程序中有多个顺序流在执行. 进程:每个进程都有独立的代码和数据空间(进程上下文),进程间的切换会有较大的开销,一个进程包含1--n个线程. 线程:同一类线 ...
- unity3d AssetStore 下载的资源位置
Win7,8和Win10系统下: 进入C:\Users\“用户名”目录,然后打开文件夹选项的显示隐藏文件选项 再进入AppData\Roaming\Unity/Asset Store-5.x下找到 M ...
- java线程基础知识----java daemon线程
java线程是一个运用很广泛的重点知识,我们很有必要了解java的daemon线程. 1.首先我们必须清楚的认识到java的线程分为两类: 用户线程和daemon线程 A. 用户线程: 用户线程可以简 ...