剑指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在数组 ...
随机推荐
- 对MySQL数据类型的认识
简述 良好的逻辑设计和物理设计是高性能系统的基石,比如反范式设计可以加快某些类型的查询同时也会影响另外一些类型的查询效率,所以我们必须重视Mysql对于数据库的设计(本文主要讲述表字段类型对于数据库性 ...
- 优先队列 STL (转)
优先队列是队列的一种,不过它可以按照自定义的一种方式(数据的优先级)来对队列中的数据进行动态的排序每次的push和pop操作,队列都会动态的调整,以达到我们预期的方式来存储. 例如:我们常用的操作就是 ...
- cnetos升级内核玩docker
最近在学习docker容器.在阿里云上的服务器内核版本比较低.所以,需要先升级. 查看内核命令:uname -r 升级内核,网上也有很多种方式.一般都是下载内核包,然后自己编译.不过这种方式需要注意的 ...
- IE开发人员工具手册
The DOM Explorer tool (CTRL + 1) The The DOM Explorer tool shows the structure of your webpage as it ...
- 【Spark SQL 源码分析系列文章】
从决定写Spark SQL源码分析的文章,到现在一个月的时间里,陆陆续续差不多快完成了,这里也做一个整合和索引,方便大家阅读,这里给出阅读顺序 :) 第一篇 Spark SQL源码分析之核心流程 第二 ...
- Linux系统官网下载
CentOS-6.9-x86_64-bin-DVD1.isohttp://archive.kernel.org/centos-vault/6.9/isos/x86_64/CentOS-6.9-x86_ ...
- Android获取所有应用的资源id和对应的uri
背景 在某些应用中,为了实现应用apk资源放入重复利用,或者使用反射得到本应用的资源,需要使用反射方式获得,但Resources类中也自带了这种获取方式,并且功能更加强大 你可以获取string,co ...
- yii2:redis调用
参照手册,调用redis,报错,真坑: Yii::$app->redis 后改改用: Yii::getRedis();
- Docker 应用实例
Docker安装Nginx 方法一.通过 Dockerfile构建 创建Dockerfile 首先,创建目录nginx,用于存放后面的相关东西. runoob@runoob:~$ mkdir -p ~ ...
- python 超时异常处理
异常处理具体见:[循序渐进学Python]9.异常处理 环境平台:Python2.7.9 + Win8.1 本篇记录一下自己写爬虫的遇到的问题,程序中批量获取图片地址,然后批量保存.由于没有设置网址打 ...