Java 数组高频考点分析讲解
作者:Scintillator. 发布时间:2021-09-01 13:14:36
1、数组理论基础
数组是存放在连续内存空间上的相同类型数据的集合,可以通过下标索引的方式获取到下标下对应的数据。
举个栗子(字符数组)~
可以看到:
1、数组的下标从0开始
2、数组在内存中的地址是连续的
所以在删除元素时,只能用覆盖的方式进行。
例如,要删除下标为2的元素~ 就需要将从2之后的元素依次移到前一个,覆盖掉要删除的元素。
所以删除元素并不是将该元素的空间释放了,而是将后面的元素移到前面,覆盖掉要删除的元素,然后将数组的长度减去1,就能得到一个看似新的数组。
在java中,二维数组的存储方式如下:
2、常见考点
1.二分查找
力扣题目链接: 二分查找
这道题目的前提是有序数组,因为一旦有重复元素,使用二分查找法返回的元素下标可能不是唯一的,这些都是使用二分法的前提条件。
二分查找思想是:
数组有序的前提下(假设升序),如果数组中间的值大于要查找的值,那么要查找的元素就不可能在后半部分,因为后半部分的值都大于中间的值,所以通过第一次比较,就可以将范围缩小一半,后面同理,即时间复杂度降到了O(logN),效率大大提高,当题目中要求查找元素的时间复杂度为O(logN)时,首先想一想是否能用二分呢?
class Solution {
public int search(int[] nums, int target) {
// 避免当 target 小于nums[0] nums[nums.length - 1]时多次循环运算
if (target < nums[0] || target > nums[nums.length - 1]) {
return -1;
}
int left = 0, right = nums.length - 1;
while (left <= right) {
int mid = left + ((right - left) >> 1);
if (nums[mid] == target)
return mid;
else if (nums[mid] < target)
left = mid + 1;
else if (nums[mid] > target)
right = mid - 1;
}
return -1;
}
}
2.移除元素
有的同学可能说了,多余的元素,删掉不就得了?但是要知道数组的元素在内存地址中是连续的,不能单独删除数组中的某个元素,只能覆盖。
例如:给你一个数组和一个val值,要求删除数组中等于val值的元素,怎么做呢?
思路1:暴力法
我们可以使用两个for循环,当遍历到等于val值的元素时,就将后面的元素整体往前移一个覆盖掉要删除的元素,但是这种做法显然时间复杂度太高。
class Solution {
public int removeElement(int[] nums, int val) {
int size = nums.length;
for (int i = 0; i < size;i++ ) {
if (nums[i] == val) { // 发现需要移除的元素,就将数组后面集体向前移动一位
for (int j = i + 1; j < size; j++) {
nums[j - 1] = nums[j];
}
i--; // 因为下标i以后的数值都向前移动了一位,所以i也向前移动一位
size--;
}
}
return size;
}
}
思路2:双指针法
分别设设一个快慢指针,slow fast ,两者一起走,当慢指针遇到要删除的元素时停下,等待着被删除(覆盖);当快指针走到要被留下的元素时,将快指针的元素赋值给慢指针,然后两指针同时向后走,直到快指针遍历完整个数组。
可以这么理解:定义数组的新长度newLength ,从0开始,定义一个快指针遍历数组 fast,当fast走到要被留下的元素时,说明该元素应该被添加到新数组中(即被添加到newLength 下标,这里相当于 newLength 之前的部分数组看做要返回的新数组,相当于往这个新数组里插入元素)。
class Solution {
public int removeElement(int[] nums, int val) {
int fast = 0;// 定义一个快指针遍历数组
int newLength = 0;// 定义新的数组长度
while(fast < nums.length){
if(nums[fast] != val){
nums[newLength++] = nums[fast];
}
fast++;
}
return newLength;
}
}
推荐力扣题目
1.删除排序数组中的重复项
2.移动零
3.比较含退格的字符串
4.有序数组的平方
其他常见数组的考点很多都是以这两点为基础,无非就是对数组增删改查,将数组的查找和删除掌握了,就可以开始刷题啦。
来源:https://blog.csdn.net/qq_45792749/article/details/123672348


猜你喜欢
- 在网上很多关于dubbo异常统一处理的博文,90%都是抄来抄去。大多都是先上一段dubbo中对于异常的统一处理的原码,然后说一堆的(甚至有1
- package GraphicsCanvas;import java.awt.BorderLayout;import java.awt.Ca
- 本文实例讲述了Java使用Preference类保存上一次记录的方法。分享给大家供大家参考。具体分析如下:在使用java中JFileChoo
- 一、 四种修饰符的说明public修饰符表示 公有 。此修饰符的范围最大。当不声明任何修饰符时,系统会默认使用此修饰符。internal修饰
- 本文实例讲述了C#使用委托实现的快速排序算法。分享给大家供大家参考。具体如下:class QuickSort { private dele
- 当游戏在手机/模拟器上卡死,logcat没有日志输出,也没有卡死堆栈信息或者bugly也没有捕获到异常,你是否很焦急?本文介绍一下我们项目中
- gRPCgRPC是由 google开发的一个高性能、通用的开源RPC框架,主要面向移动应用开发且基于HTTP/2协议标准而设计,同时支持大多
- MyEclipse2017创建Spring项目,供大家参考,具体内容如下1、创建一个Web Project2、右击项目-->Prope
- 一、简介1.什么是GUID?全局唯一标识符(GUID,Globally Unique Identifier),GUID也称作 UUID(Un
- 一.类与接口的区别类:描述了一个实体,包括实体的状态,也包括实体可能发出的动作。接口:定义了一个实体可能发出的动作。但是只是定义了这些动作的
- 今天遇到一个问题,原来用的好好的asp网页无法打开,同时inetinfo.exe的CPU占用率很高,几乎达到了100%。看了一下系统错误日志
- 如果对共享的可变数据的访问不能同步,其后果非常可怕,即使这个变量是原子可读写的。下面考虑一个线程同步方面的问题。对于线程同步,Java类库提
- 本文实例讲述了简单记事本java实现代码。分享给大家供大家参考。具体如下:完整代码如下:import java.awt.*;import j
- Android屏蔽软键盘并且显示光标的实例详解如果是android4.0以下,那么editText.setInputType(InputTy
- Android 调用百度地图API一、到 百度地图开发平台下载SDKhttp://lbsyun.baidu.com/index.php?ti
- 可以给已有实体类动态的添加字段并返回新的实体对象,不影响原来的实体对象结构。添加依赖<dependency> &n
- C++ 中二分查找递归非递归实现并分析二分查找在有序数列的查找过程中算法复杂度低,并且效率很高。因此较为受我们追捧。其实二分查找算法,是一个
- 微信的发送语音是有一个向上取消的,我们使用onTouchListener来监听手势,然后做出相应的操作就行了。直接上代码://语音操作对象p
- 1、多个线程对同一个队列进行读写操作,要注意进行读写控制,某个线程在读取的时候,不允许其它线程读、写;某个线程在写的时候,不允许其它线程进行
- 最近做了一个项目其中有需求,要实现自动登录功能,通过查阅相关资料,打算用session监听来做,下面给大家列出了配置 * 的方法:1.在项目