剑指Offer:面试题29——数组中出现次数超过一半的数字(java实现)
PS:在前几天的面试中,被问到了这个题。然而当时只能用最低效的方法来解。
问题描述:
数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。
思路1:
低效的做法:直接用哈稀表来存储
<key, value>,并找出出现次数value大于等于一半的那个key。
代码:
import java.util.Map;
import java.util.HashMap;
public class Solution {
boolean InvalidInput = false;
public int MoreThanHalfNum_Solution(int [] array) {
if(array == null || array.length == 0){
InvalidInput = true;
return 0;
}
Map<Integer, Integer> map = new HashMap<Integer, Integer>();
int max = -1;
int ind = -1;
for(int i = 0; i < array.length; i++){
if(!map.containsKey(array[i])){
map.put(array[i], 1);
}else{
map.put(array[i], map.get(array[i])+1);
}
if(max < map.get(array[i])){
max = map.get(array[i]);
ind = array[i];
}
}
if(max > array.length/2){
return ind;
}else{
return 0;
}
}
}
思路2:
—由于出现的次数超过了数组的一半,那么如果我们将数组排序,中间的那个数一定是要求的数字(前提是它一定存在)
—这里排序时采用快排的基本思想,随机选择一个数,调整数组使得它左边的数都小于它,右边的数都大于它。(参考Partition函数)(下一篇博客的代码中给出了关于Partition函数详细的注释)
import java.util.Map;
import java.util.HashMap;
public class Solution {
static boolean InvalidInput = false;
public int MoreThanHalfNum_Solution(int [] array) {
if(array == null || array.length == 0){
InvalidInput = true;
return 0;
}
int middle = array.length >> 1;//设定中位数位置
int start = 0;
int end = array.length - 1;
int index = Partition(array, start, end);//随机选择一个标杆,进行调整数组
while(index != middle){//若所选的标杆最后位置不是中位数的位置,则根据情况对符合情况的部分进行再次调整
if(index > middle){
end = index - 1;
index = Partition(array, start, end);
}else{
start = index + 1;
index = Partition(array, start, end);
}
}
int result = array[middle];
//检查中位数出现的次数是否超过了一半
if(!CheckMoreThanHalf(array, result)){
result = 0;
}
return result;
}
public static boolean CheckMoreThanHalf(int[] array, int number){
int times = 0;
int length = array.length;
for(int i = 0; i < length; i++){
if(array[i] == number){
times++;
}
}
boolean isMoreThanHalf = true;
if(times * 2 <= length){
InvalidInput = true;
isMoreThanHalf = false;
}
return isMoreThanHalf;
}
public static int Partition(int[] array, int start, int end){
if(array == null || array.length == 0 || start < 0 || end < 0){
InvalidInput = true;
return -1;
}
int index = RandomInRange(start ,end);
int tmp = array[index];
array[index] = array[end];
array[end] = tmp;
int small = start - 1;
for(index = start; index < end; index++){
if(array[index] < array[end]){
small++;
if(small != index){
int temp = array[index];
array[index] = array[small];
array[small] = temp;
}
}
}
small++;
tmp = array[small];
array[small] = array[end];
array[end] = tmp;
return small;
}
public static int RandomInRange(int s, int e){
int Min = s;
int Max = e;
int result = Min + (int)(Math.random() * ((Max - Min) + 1));
return result;
}
}
思路3:
根据数组特点找出O(n)的算法
1.数组中有一个数字出现的次数超过数组长度的一半,也就是说它出现的次数比其他所有的数字出现的次数的和还要多。
2.因此我们可以考虑在遍历数组的时候保存两个值:一个是数组中的一个数字,一个是次数。
3.当我们遍历到下一个数字的时候,如果下一个数字和我们之前保存的数字相同,则次数加1,否则减1.
代码:
import java.util.Map;
import java.util.HashMap;
public class Solution {
static boolean InvalidInput = false;
public int MoreThanHalfNum_Solution(int [] array) {
if(array == null || array.length == 0){
InvalidInput = true;
return 0;
}
int result = array[0];
int times = 1;
for(int i = 1; i < array.length; i++){
if(times == 0){
result = array[i];
}else if(array[i] == result){
times++;
}else{
times--;
}
}
if(!CheckMoreThanHalf(array, result)){
InvalidInput = true;
result = 0;
}
return result;
}
public static boolean CheckMoreThanHalf(int[] array, int number){
int times = 0;
int length = array.length;
for(int i = 0; i < length; i++){
if(array[i] == number){
times++;
}
}
boolean isMoreThanHalf = true;
if(times * 2 <= length){
InvalidInput = true;
isMoreThanHalf = false;
}
return isMoreThanHalf;
}
}
PS:检查出现频率最高的是否超过一半,也还是需要遍历一次数组的。
剑指Offer:面试题29——数组中出现次数超过一半的数字(java实现)的更多相关文章
- 剑指Offer - 九度1370 - 数组中出现次数超过一半的数字
剑指Offer - 九度1370 - 数组中出现次数超过一半的数字2013-11-23 03:55 题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组 ...
- 【剑指Offer】28、数组中出现次数超过一半的数字
题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 例如:输入如下所示的一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过 ...
- 《剑指offer》39题—数组中出现次数超过一半的数字
题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...
- 剑指Offer:找出数组中出现次数超过一半的元素
题目:找出数组中出现次数超过一半的元素 解法:每次删除数组中两个不同的元素,删除后,要查找的那个元素的个数仍然超过删除后的元素总数的一半 #include <stdio.h> int ha ...
- 剑指offer(28)数组中出现次数超过一半的数
题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. ...
- 数组中出现次数超过一半的数字 -java
数组中出现次数超过一半的数字 -java 方法一: 数组排序,然后中间值肯定是要查找的值. 排序最小的时间复杂度(快速排序)O(NlogN),加上遍历. 方法二: 使用散列表的方式,也就是统计每个数组 ...
- 剑指Offer:面试题15——链表中倒数第k个结点(java实现)
问题描述 输入一个链表,输出该链表中倒数第k个结点.(尾结点是倒数第一个) 结点定义如下: public class ListNode { int val; ListNode next = null; ...
- 【Offer】[39] 【数组中出现次数超过一半的数字】
题目描述 思路分析 测试用例 Java代码 代码链接 题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如,输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数 ...
- 《剑指offer》面试题39. 数组中出现次数超过一半的数字
问题描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 你可以假设数组是非空的,并且给定的数组总是存在多数元素. 示例 1: 输入: [1, 2, 3, 2, 2, 2, 5, 4, ...
随机推荐
- UITabBarButton 点击失效问题
开发过程: 在创建一个UIWindow时,直接在window上添加手势动作. 开发代码: UITapGestureRecognizer *tapRecognizer=[[UITapGestureRec ...
- bzoj1188 [HNOI2007]分裂游戏 博弈论 sg函数的应用
1188: [HNOI2007]分裂游戏 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 973 Solved: 599[Submit][Status ...
- 2016HUAS_ACM暑假集训4C - 递推
题目大意:给你一个高为n ,宽为m列的网格,计算出这个网格中有多少个矩形. 这里我们可以这样看: 对于行:假如是m单位长度,则长度为1的矩形有m个,长度为2的矩形有m-1个......长度为m的只有1 ...
- git tool
1. install apt-get install git 2. 使用 ssh-keygen -C "email" -t rsa 生成一个ssh pub key,将生成的id_ ...
- Mysql(一)
一.Mysql简介 Mysql是一个关系型数据库管理系统,由瑞典MySQL AB公司开发,目前属于Oracle旗下.特点:其体积小.速度快.开源. 分为社区办和商业版,其社区版性能卓越. 二.Ubun ...
- solr6.0.0 + tomcat8 配置问题
中间件需求: apache-tomcat-8.0.32.tar.gz jdk-8u74-linux-x64.rpm solr-6.0.0.zip 0.安装java JDK rpm -ivh jdk-8 ...
- install Hadoop
Installing Java Hadoop runs on both Unix and Windows operating systems, and requires Java to beinsta ...
- Oracle索引碎片检查及定期重建常用表的索引
背景说明: 今天查阅书籍时,偶然间发现“在对某个索引行执行删除操作时,只是为该行增加了一个删除标记,这个索引行并不会释放它的存储空间,Insert产生的新的索引行也不能被插入到该位置.索引列的修改过程 ...
- 【linux】 解决linux下vsftp 500 OOPS: cannot change directory:/home/ftp/ 办法
用FileZilla连接ftp出现错误,500 OOPS: cannot change directory:/home/ftp 原因是CentOS系统安装了SELinux,因为默认下是没有开启FTP的 ...
- Selenium2入门(一)简介
Selenium 是一款有名的 Web 应用程序测试框架,用于进行功能测试.新版本 Selenium 2 结合了 Selenium 1 和 WebDriver(Selenium 的并行项目)中的最佳特 ...