剑指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在数组 ...
随机推荐
- 分布式集群Session原理及实现共享
1.什么是Session/Cookie? 用户使用网站的服务,基本上需要浏览器与Web服务器的多次交互.HTTP协议本身是无状态的,当用户的第一次访问请求结束后,后端服务器就无法知道下一次来访问的还是 ...
- Sqoop-将Hive ORC表导出到MySQL
Sqoop-将Hive ORC表导出到MySQL sqoop export --connect jdbc:mysql://localhost:3306/test --username root --p ...
- AIDL与Binder的区别
Binder是一个远程对象的基础类,核心部分是远程调用机制,这部分是由IBinder定义的. 它是对IBinder类的实现,其中IBinder类提供了这样一个类的标准的本地化实现方式. 大多数开发者不 ...
- activity启动模式之singleTask
activity启动模式之singleTask 一.简介 如果另外一个应用调用了C2,C2在栈底,如果这个程序里面再嗲用C1,C3,C2,那么这个C2就是调用onNewIntant的,C1和C3都被销 ...
- spring3: Aspectj后置返回通知
Aspectj后置返回通知 接口: package chapter1.server; public interface IHelloService2 { public int sayAfterRetu ...
- C++(二十三) — 内存泄漏及指针悬挂
1.内存泄漏 动态申请的内存空间没有正常释放,但也不能继续使用. ; pch1 = new char('A'); // 此处申请的空间未被释放. char *pch2 = new char; pch1 ...
- web自动化常用定位和方法总结
一. driver常用方法 二. 常用定位 三. 元素在页面不可见区域 四. iframe的操作 五. 页面弹出框:加等待时间 六. windows弹出框 七. 鼠标操作 八. 下拉列表 注意:下图中 ...
- "下载"文件夹的desktop.ini
下载 [.ShellClassInfo] LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21798 IconResource=%S ...
- 解决Oracle数据库IP地址改变创建数据库的问题
方案一 查了相关的资料后才知道,只要修改安装Oracle的目录下面的listener.ora和tnsnames.ora里面的内容即可.看图看图 修改什么内容呢?看图看图. 就是HOST地址改一下就可 ...
- FMDB官方使用文档 G-C-D的使用 提高性能(翻译)
由于FMDB是建立在SQLite的之上的,所以你至少也该把这篇文章从头到尾读一遍.与此同时,把SQLite的文档页 加到你的书签中.自动引用计数(APC)还是手动内存管理呢? 两种都行,FMDB会 ...