LeetBook 数组题目 java实现

简介: LeetBook 数组题目 java实现

1.删除排序数组的重复项

题目:


d22b4936188f4c5bb38d816b70c79f74.png

解题思路:

代码:

public int removeDuplicates(int[] nums) {
    //当数组长度为0时,直接返回零即可
    if (nums.length == 0) {
        return 0;
    }
    //如果数组长度大于0,则返回的数组长度至少为1,记录数先保存所存入的第一个元素
    int record = nums[0];
    int num = 1;
    //直接从数组的第二个元素开始判断
    for (int i = 1; i < nums.length; i++) {
        //如果所遍历到的新元素不等于记录数所保存的值,说明出现的是新元素,则更新记录数,并将该数存储到所给数组前面,num即当前找到的第几个不重复的数
        if (nums[i] != record) {
            nums[num]=nums[i];
            num++;
            record=nums[i];
        }
    }
    return num;
}


//官方做法
public int removeDuplicates(int[] nums) {
    int n = nums.length;
    if (n == 0) {
        return 0;
    }
    //slow用于记录当前找到的不重复数的数量,当数组长度大于0时,最起码有一个或以上的不重复的数
    int fast = 1, slow = 1;
    while (fast < n) {
        //找到不重复的数,存储到数组的第slow个元素中
        if (nums[fast] != nums[fast - 1]) {
            nums[slow] = nums[fast];
            ++slow;
        }
        ++fast;
    }
    return slow;
}

2.买卖股票的最佳时机II

题目:f64df0336e644dbebd489f6d0edef67f.png


解题思路:允许多次买卖,意味着所有的差价都可以赚到,所以题目就相当于求解可以转到的最大总差价

代码:

public class Solution {
    public int maxProfit(int[] prices) {
        if(prices.length==1){
            return 0;
        }
        int reslut = 0;
        int i = 0;
        int j = 1;
        while (j<prices.length){
            int p1 = prices[i];
            int p2 = prices[j];
            if(p2>=p1){
                reslut += (p2-p1);
            }
            i++;
            j++;
        }
        return reslut;
    }
}


class Solution {
    public int maxProfit(int[] prices) {
        int sum=0;
        //无论数组长度为0还是1,都无法赚钱
        if(prices.length<=1){
            return 0;
        }
        for(int i=1;i<prices.length;i++){
            if(prices[i]>prices[i-1]){
                sum+=prices[i]-prices[i-1];
            }
        }
        return sum;
    }
}

3.旋转数组


6f4b60191c4946e2b2f01e4d7e95489d.png


解题思路:由于旋转数组具有周期性,所以可以采用取余法获取旋转次数,而后进行模拟,这样可以大量减少模拟次数

代码:

public voidpublic void rotate(int[] nums, int k) {
    int length = nums.length;
    k = k%length;
    if(k!=0){
        int firstIndex = k;
        int[] arr = nums.clone();
        for (int i = 0; i < arr.length ; i++) {
            if(firstIndex>=nums.length){
                firstIndex = 0;
            }
            nums[firstIndex] = arr[i];
            firstIndex++;
        }
    }
}


4.存在重复元素7da874d4df2f4a57adfb41dd8ca4f8cc.png

代码:

/**
 * 解题思路:我们知道set集合中的元素是不能有重复的,在添加的时候如果有重复的,会把之前的值给覆盖,并且返回false。
 * 我们遍历数组中的所有值,一个个添加到集合set中,在添加的时候如果返回false,就表示有重复的,直接返回true。
 */
public class Solution {
    public boolean containsDuplicate(int[] nums) {
        Set<Integer> set = new HashSet<>();
        for (int num : nums) {
            //因为集合set中不能有重复的元素,如果有重复的
            //元素添加,就会添加失败
            if (!set.add(num))
                return true;
        }
        return false;
    }
}


5.只出现一次的数字


代码:

/**
 * 解题思路:位运算解决
 * 这题说的是只有一个数出现了一次,其他数字都出现了2次,让我们求这个只出现一次的数字。这题使用位运算是最容易解决的,关于位运算有下面几个规律
 * 1^1=0;
 * 1^0=1;
 * 0^1=1;
 * 0^0=0;
 * 也就说0和1异或的时候相同的异或结果为0,不同的异或结果为1,根据上面的规律我们得到
 * a^a=0;自己和自己异或等于0
 * a^0=a;任何数字和0异或还等于他自己
 * a^b^c=a^c^b;异或运算具有交换律
 * 有了这3个规律,这题就很容易解了,我们只需要把所有的数字都异或一遍,最终的结果就是我们要求的那个数字。来看下代码
 */
public class Solution {
    public int singleNumber(int[] nums) {
        int reduce = 0;
        for (int num : nums) {
            reduce =  reduce ^ num;
        }
        return reduce;
    }
}

6.两个数组的交集 II


0535736357704852b6a8f5f52fb95a2a.png


代码:

public int[] intersect(int[] nums1, int[] nums2) {
    HashMap<Integer, Integer> map = new HashMap<>();
    for (int i = 0; i < nums1.length; i++) {
        if (!map.containsKey(nums1[i])) {
            map.put(nums1[i], 1);
        } else {
            map.replace(nums1[i], map.get(nums1[i]) + 1);
        }
        //或者使用以下方式
        //map.put(nums1[i], map.getOrDefault(nums1[i], 0) + 1);
    }
    List<Integer> numList = new ArrayList<>();
    for (int num2 : nums2) {
        //当Map集合中存在这个key时,就使用这个key值,(若是数值型可以在此基础上进行运算)
        //如果没有就使用默认值defaultValue,即所设置的0
        if (map.getOrDefault(num2,0)>0){
            numList.add(num2);
            map.put(num2, map.get(num2) - 1);
        }
    }
    int[] arr = new int[numList.size()];
    for (int i = 0; i < numList.size(); i++) {
        arr[i]=numList.get(i);
    }
    return arr;
}


7.加一a65bfe52245d44fc9fcfa427c4a4ee89.png


代码:

public int[] plusOne(int[] digits) {
    //用来记录一个位数是否要执行加1操作,比如个位数是9,则十位数需要执行加一操作
    int count = 1;
    for (int i = digits.length - 1; i >= 0; i--) {
        //大于9,下一位数还需要加一
        if (digits[i] + count > 9) {
            digits[i] = 0;
            count = 1;
        } else {
            //小于等于9,下一位数无须再加一,count=0
            digits[i] = digits[i] + count;
            count = 0;
        }
    }
    //当最后count==1时,说明数组长度不够,如999>1000
    if (count == 1) {
        int[] arr = new int[digits.length + 1];
        //只有首位是1
        arr[0] = 1;
        return arr;
    } else {
        return digits;
    }
}



8.移动零

代码:

public void moveZeroes(int[] nums) {
    int length = nums.length;
    if (length <= 1) {
        return;
    }
    int fast = 0;
    int slow = 0;
    while (fast < length) {
        //不为零地数往前移动
        if (nums[fast] != 0) {
            nums[slow] = nums[fast];
            slow++;
        }
        fast++;
    }
    //后面的数置为0
    while (slow<length){
        nums[slow++]=0;
    }
}


9.两数之和


代码:

//暴力搜索
public int[] twoSum(int[] nums, int target) {
    int[] answer=new int[2];
    int length=nums.length;
    for(int i=0;i<length-1;i++){
        for(int j=i+1;j<length;j++){
            if(nums[i]+nums[j]==target){
                answer[0]=i;
                answer[1]=j;
                return answer;
            }
        }
    }
    return answer;
}


//优化之后的方法
public int[] twoSum(int[] nums, int target) {
    //存放元素及其index
    HashMap<Integer, Integer> map = new HashMap<>();
    for (int i = 0; i < nums.length; i++) {
        if (map.getOrDefault(target - nums[i], -1) > -1) {
            return new int[]{map.get(target - nums[i]), i};
        }
        map.put(nums[i],i);
    }
    return null;
}


总结

提示:这里对文章进行总结:

数组是数据结构的入门,虽然看似简单,但也存在很多操作技巧,感兴趣的同学可以上初级算法 - LeetBook - 力扣(LeetCode)全球极客挚爱的技术成长平台 (leetcode-cn.com)进行进一步的学习。

目录
相关文章
|
3月前
|
Java
Java 数组学习笔记
本文整理Java数组常用操作:遍历、求和、查找、最值及二维数组行求和等典型练习,涵盖静态初始化、元素翻倍、去极值求平均等实例,帮助掌握数组基础与应用。
|
4月前
|
存储 缓存 Java
Java数组全解析:一维、多维与内存模型
本文深入解析Java数组的内存布局与操作技巧,涵盖一维及多维数组的声明、初始化、内存模型,以及数组常见陷阱和性能优化。通过图文结合的方式帮助开发者彻底理解数组本质,并提供Arrays工具类的实用方法与面试高频问题解析,助你掌握数组核心知识,避免常见错误。
|
5月前
|
存储 Java 索引
java 数组
在 Java 中,数组是一种数据结构,用于存储多个相同类型的数据元素。数组的大小一旦创建后就不能改变,因此它是固定长度的。Java 数组是一种 对象,即使它存储的值是基本类型(如 int、double 等),它也是一个对象引用。
151 0
|
6月前
|
缓存 NoSQL Java
Java Redis 面试题集锦 常见高频面试题目及解析
本文总结了Redis在Java中的核心面试题,包括数据类型操作、单线程高性能原理、键过期策略及分布式锁实现等关键内容。通过Jedis代码示例展示了String、List等数据类型的操作方法,讲解了惰性删除和定期删除相结合的过期策略,并提供了Spring Boot配置Redis过期时间的方案。文章还探讨了缓存穿透、雪崩等问题解决方案,以及基于Redis的分布式锁实现,帮助开发者全面掌握Redis在Java应用中的实践要点。
357 6
|
6月前
|
算法 Java 关系型数据库
校招 Java 面试基础题目解析及学习指南含新技术实操要点
本指南聚焦校招Java面试,涵盖Java 8+新特性、多线程与并发、集合与泛型改进及实操项目。内容包括Lambda表达式、Stream API、Optional类、CompletableFuture异步编程、ReentrantLock与Condition、局部变量类型推断(var)、文本块、模块化系统等。通过在线书店系统项目,实践Java核心技术,如书籍管理、用户管理和订单管理,结合Lambda、Stream、CompletableFuture等特性。附带资源链接,助你掌握最新技术,应对面试挑战。
156 2
|
6月前
|
存储 算法 Java
校招 java 面试基础题目及解析
本文围绕Java校招面试基础题目展开,涵盖平台无关性、面向对象特性(封装、继承、多态)、数据类型、关键字(static、final)、方法相关(重载与覆盖)、流程控制语句、数组与集合、异常处理等核心知识点。通过概念阐述和代码示例,帮助求职者深入理解并掌握Java基础知识,为校招面试做好充分准备。文末还提供了专项练习建议及资源链接,助力提升实战能力。
156 0
|
6月前
|
安全 Java 编译器
Java 校招面试题目合集及答案 120 道详解
这份资料汇总了120道Java校招面试题目及其详细答案,涵盖Java基础、JVM原理、多线程、数据类型、方法重载与覆盖等多个核心知识点。通过实例代码解析,帮助求职者深入理解Java编程精髓,为校招面试做好充分准备。无论是初学者还是进阶开发者,都能从中受益,提升技术实力和面试成功率。附带的资源链接提供了更多学习材料,助力高效备考。
295 3
|
7月前
|
存储 人工智能 Java
打乱数组内容引发的问题( Java)
本文介绍了两种实现数组随机打乱的方法,并深入探讨了Java中原始数据类型与对象类型的差异。方法一通过自定义随机数交换数组元素位置,方法二借助`Collections.shuffle()`函数完成数组打乱。同时,文章详细解析了`int`和`Integer`的区别,包括声明方式、内存占用、初始化以及对象特性等,并讲解了自动装箱与拆箱的功能,帮助读者更好地理解Java的基础知识。
129 0
|
8月前
|
存储 Java 数据挖掘
Java 中数组的多种定义方式
本文深入解析了Java中数组的多种定义方式,涵盖基础的`new`关键字创建、直接初始化、动态初始化,到多维数组、`Arrays.fill()`方法以及集合类转换为数组等高级用法。通过理论与实践结合的方式,探讨了每种定义方法的适用场景、优缺点及其背后的原理,帮助开发者掌握高效、灵活的数组操作技巧,从而编写更优质的Java代码。
409 0