剑指offer-第五章优化时间和空间效率(数组中出现次数超过一半的数字)
题目:输入一个数组,找出一个数字,它在数组中出现的次数超过数组的一半。
题目规定如果可以改变数组中元素的位置。
思路1:如果数组是排序的,那么中间元素的位置不就是次数超过数组一半的元素吗?是的,因此我们可以才用partition来做判断。如果partition的得到的整数index是在数组的中间,那么该元素就是超过数组中一半次数的元素。时间效率为O(n)。
题目规定不可以改变元素的位置
思路2:此时,我们不能对数据进行排序,因此,我们可以考虑到数组中出现次数超过一半的元素比其他所有的元素都出现的次数多,因此我们可以用一个记录数据出现次数的times来记录数组元素出现的次数。如果该元素出现times加一,如果下一个元素不是该元素,times减一。那么最后一个times设置为1的一定是超过数组一半次数的元素。
Java代码:思路1
import java.util.Random; //允许改变数组中元素的位置,找到超过数组一半的元素。
public class HalfOfArrary {
//如果数组是排序的,找超过一半的数,就很好找,因此我们先来一个快排
public int partition(int[] a,int start,int end){
if(a==null)
return-1;
Random rand=new Random();
int index=start+rand.nextInt(end-start);
swrap(a,index,end);
int small=start-1;
for(index=start;index<end;index++){
if(a[index]<a[end]){
small++;
if(small!=index)
swrap(a,small,index);
}
}
small++;
swrap(a,small,end);
return small;
} public void swrap(int[] a, int index, int end) {
int temp=a[index];
a[index]=a[end];
a[end]=temp;
}
//查找超过一半的数,并返回
public int moreThanHalf(int[] a){
if(a==null)
return 0;
int start=0;
int len=a.length;
int middle=len>>1;
int end=len-1;
int index=partition(a,start,end);
while(index!=middle){
if(index>middle)
{
end=index-1;
index=partition(a,start,end);
}
else{
start=index+1;
index=partition(a,start,end);
}
}
int result=a[middle];
if(!(isMoreThanHalf(a,result))){
return 0;
}
return result; }
//判断数组中是否存在,超过一半的数
public boolean isMoreThanHalf(int[] a, int result) {
boolean isMoreHalf=true;
int time=0;
if(a==null)
return false;
for(int i=0;i<a.length;i++){
if(a[i]==result)
time++;
}
int middle=a.length>>1;
if(time>middle)
isMoreHalf=true;
return isMoreHalf; }
public static void main(String[] args){
int[] a={1,1,1,5,3,4,3,1,1,1,1};
HalfOfArrary hoa=new HalfOfArrary();
int result=hoa.moreThanHalf(a);
System.out.println(result+" ");
}
}
Java代码:思路2
//当规定不能改变数组中元素的位置的时候,找到大于数组中一半数的元素
public class MoreThanOfHalf {
public int moreThanHalfOfArray(int[] a){
if(a==null)
return 0;
int times=1;//标记数组中元素次数
int result=a[0];
for(int i=1;i<a.length;i++){
if(times==0){
result=a[i];
times=1;
}
else if(times==result)
times++;
else
times--;
}
if(!(isMoreThanHalf(a,result)))
result=0;
return result;
}
//判断数组中是否存在,超过一半的数
public boolean isMoreThanHalf(int[] a, int result) {
boolean isMoreHalf=true;
int time=0;
if(a==null)
return false;
for(int i=0;i<a.length;i++){
if(a[i]==result)
time++;
}
int middle=a.length>>1;
if(time>middle)
isMoreHalf=true;
return isMoreHalf; }
public static void main(String[] args){
int[] a={1,2,3,3,4,3,4,3,3,3};
MoreThanOfHalf mh=new MoreThanOfHalf();
int result=mh.moreThanHalfOfArray(a);
System.out.println(result+"");
}
}
剑指offer-第五章优化时间和空间效率(数组中出现次数超过一半的数字)的更多相关文章
- 剑指Offer面试题29(java版):数组中出现次数超过一半的数字
题目:数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 比如输入一个长度为9的数组{1,2,3,2.2,2.5,4,2}.因为数字2在数组中出现5次,超过数组长度的一半,因此输出2. 解 ...
- 剑指offer-第五章优化时间和空间效率(从1到n的整数中1出现的次数)
题目:输入一个整数n,从1到n这n个十进制整数中1出现的次数. 思路1:对1到n中的任意一个数i对其进行求余数来判断个位是否为1,然后再求除数,判断十位是否为1.统计出1的个数.然后对1到n用一个循环 ...
- 剑指offer-第五章优化时间和空间效率(数组中的逆序对的总数)
题目:在数组中如果两个数字的前面的数比后面的数大,则称为一对逆序对.输入一个数组求出数组中逆序对的总数. 以空间换时间:思路:借助一个辅助数组,将原来的数组复制到该数组中.然后将该数组分成子数组,然后 ...
- 剑指offer-第五章优化时间和空间效率(把数组排列成最小的数)
题目:输入一个正整数数组,将所有的数,排列起来,组成一个最小的数.
- 剑指offer-第五章优化时间和空间效率(两个链表的第一个公共节点)
思路1:要求的是两个链表的第一个公共节点,首先想到的是用栈来存放两个链表,然后依次从栈中抛出,直到最后一个相同的节点为止.但是要用到两个栈,空间复杂度为O(n): 思路2:从头到尾分别遍历两个链表得到 ...
- 剑指offer-第五章优化时间和空间效率(在字符串中第一次出现切只出现一次的字符)
题目:在字符串中第一次出现切只出现一次的字符 思路:用HashMap来存放对应的char值和该char出现的次数.做一次变量就可以得到第一个只出现一次的字符. Java代码: import java. ...
- 剑指offer-第五章优化时间和空间效率(连续子数组的最大和)
题目:输入一个数组,数组中有正也有负,数组中连续的一个或者连续的多个数字组成一个子数组.求所有的子数组和的最大值.要求时间复杂度为O(n) 思路:我们的最直观的想法就是求出这个数组中的所有的子数组,然 ...
- 剑指offer-第五章优化时间和空间效率(最小的k个数)
题目:输入n个数,输出最小的k个数. 时间复杂度为O(n) 思路1:我们想的到的最直接的思路就是对这个N个数进行排序,然后就可以找到最小的k个了,同样可以用快排partition.但是只要找到前K个最 ...
- 剑指offer第五章
剑指offer第五章 1.数组中出现次数超过一半的数 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组 ...
随机推荐
- 【c++习题】【17/4/13】stack
1.stack 模板.动态内存分配.析构 #include "stack2.cpp" #include <iostream> using namespace std; ...
- ubuntu+anaconda+python安装各版本tensorflow
一.安装anaconda 1.去官网下载anaconda linux版本即可 选择合适的版本下载即可 2.安装Aanconda: 打开终端(Ctrl+Alt+t)进入到下载的目录一般在home 下的D ...
- Oracle 数据库比较日期大小
在今天或者今天之前作比较:select * from JN_BUS_KJLWSBJBXX where dqsj < to_date('2007-09-07 00:00:00','yyyy-mm- ...
- linux下字典生成工具-crunch与rtgen
所谓的密码字典主要是配合密码破解软件所使用,密码字典里包括许多人们习惯性设置的密码.这样可以提高密码破解软件的密码破解成功率和命中率,缩短密码破解的时间.当然,如果一个人密码设置没有规律或很复杂,未包 ...
- Mac开发
工具类:
- 怎么用Python提取域名中的主域名
从一个域名里面提取主域名,初想起来,貌似很简单,不就是数点[.]的个数吗?取最后一个点前后的字符串,那 abc.txt 是域名吗?那再加个验证,加上国家码,.com,.cn,.org结尾的才算,那这个 ...
- 使用Mybatis整合spring时报错Injection of autowired dependencies failed;
这是无法自动注入,通过查找,我出错的原因在于配置文件的路径没有写对,在applicationContext.xml中是这样写的. <bean id="sqlSessionFactory ...
- JavaScript面向对象之Prototypes和继承
本文翻译自微软的牛人Scott Allen Prototypes and Inheritance in JavaScript ,本文对到底什么是Prototype和为什么通过Prototype能实现继 ...
- BusyIndicator using MVVM 忙碌状态指示器的的实现
ViewModel 视图模型 public abstract class ViewModelBase : INotifyPropertyChanged { private bool isbusy; p ...
- 51nod-1259-分块+dp
http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1259 1259 整数划分 V2 基准时间限制:1 秒 空间限制:1310 ...