算法训练Day7_哈希表 | LeetCode:454_四数相加 + 383. 赎金信 + 15_三数之和 +18_四数之和

LeetCode:LeetCode:454_四数相加

问题描述

找出给定升序数组nums的目标值target,如果存在返回下标,否则返回-1。

链接:

解决方案:

1.思路

二分法基本思路

2.代码实现

class Solution {
    public int search(int[] nums, int target) {
    int l = 0; 
    int r = nums.length-1;
    while(l <= r){
        int mid = (r-l)/2 + l;
        int num = nums[mid];
        if(target == num){
          return mid;
        }
         else if(target > num){
            l = mid+1;
        }
        else{
            r = mid-1;
        }
    } 
    return -1;
  }
}

3.复杂度分析

时间复杂度:如果有n个待处理数据元素,每次比较后待处理数据减半,故时间复杂度为 O l o g ( n ) Olog(n) Olog(n);

空间复杂度:

4.注意

  • 注意左闭右开,左闭右闭的情况区分;
  • 若数组长度取值范围较大,计算中间值mid的时候, m i d = l + r 2 mid=\frac{l+r}{2} mid=2l+r可能超出 int 类型的取值范围,从而导致计算错误。此时,需使用 m i d = l + r − l 2 mid=l+\frac{r-l}{2} mid=l+2rl计算中点,避免此问题。
  • 记得循环最外面的一层代码,要return -1,表示没有找到;
  • 记得二分法的使用前提是有序数组

5.疑问

LeetCode:383. 赎金信

问题描述

  • 移除给定无序有重复数组nums的指定元素val;
  • 要求只使用 O ( 1 ) O(1) O(1)的空间复杂;并原地修改数组;
  • 要求对移除元素之后的数组进行排序,并输出

解决方案:

1.思路:使用快慢指针

  • 快指针:记录新数组元素的数据值 ;实际上是for循环的一个一个移动
  • 慢指针:新数组元素的下标值

2.代码实现

class Solution {
    public int removeElement(int[] nums, int val) {
        int slow = 0;
        for(int fast=0;fast<nums.length;fast++){
            
            if (nums[fast]!=val){
                nums[slow++] = nums[fast];
            }
        }
        return slow;
    }
}

3.复杂度分析

  • 时间复杂度: 快指针在for循环中遍历,遇到val的元素跳过(相当于删除),遇到不等于val的元素,赋值给慢指针指向的“坑🕳”;
  • 空间复杂度:原地修改为 O ( 1 ) O(1) O(1)

4.疑惑思考

是否所有的双层循环都可以用快慢指针去解决呢?

LeetCode:15_三数之和

问题描述

  • 移除给定无序有重复数组nums的指定元素val;
  • 要求只使用 O ( 1 ) O(1) O(1)的空间复杂;并原地修改数组;
  • 要求对移除元素之后的数组进行排序,并输出

解决方案:

1.思路:使用快慢指针

  • 快指针:记录新数组元素的数据值 ;实际上是for循环的一个一个移动
  • 慢指针:新数组元素的下标值

2.代码实现

class Solution {
    public int removeElement(int[] nums, int val) {
        int slow = 0;
        for(int fast=0;fast<nums.length;fast++){
            
            if (nums[fast]!=val){
                nums[slow++] = nums[fast];
            }
        }
        return slow;
    }
}

3.复杂度分析

  • 时间复杂度: 快指针在for循环中遍历,遇到val的元素跳过(相当于删除),遇到不等于val的元素,赋值给慢指针指向的“坑🕳”;
  • 空间复杂度:原地修改为 O ( 1 ) O(1) O(1)

4.疑惑思考

是否所有的双层循环都可以用快慢指针去解决呢?

LeetCode:15_四数之和

问题描述

  • 移除给定无序有重复数组nums的指定元素val;
  • 要求只使用 O ( 1 ) O(1) O(1)的空间复杂;并原地修改数组;
  • 要求对移除元素之后的数组进行排序,并输出

解决方案:

1.思路:使用快慢指针

  • 快指针:记录新数组元素的数据值 ;实际上是for循环的一个一个移动
  • 慢指针:新数组元素的下标值

2.代码实现

class Solution {
    public int removeElement(int[] nums, int val) {
        int slow = 0;
        for(int fast=0;fast<nums.length;fast++){
            
            if (nums[fast]!=val){
                nums[slow++] = nums[fast];
            }
        }
        return slow;
    }
}

3.复杂度分析

  • 时间复杂度: 快指针在for循环中遍历,遇到val的元素跳过(相当于删除),遇到不等于val的元素,赋值给慢指针指向的“坑🕳”;
  • 空间复杂度:原地修改为 O ( 1 ) O(1) O(1)

4.疑惑思考

是否所有的双层循环都可以用快慢指针去解决呢?

  • 14
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值