【经典算法】LeetCode 35. 搜索插入位置(Java/C/Python3/Golang实现含注释说明,Easy)

  • 作者主页: 🔗进朱者赤的博客

  • 精选专栏: 🔗经典算法

  • 作者简介:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名

  • ❤️觉得文章还不错的话欢迎大家点赞👍➕收藏⭐️➕评论,💬支持博主,记得点个大大的
    关注,持续更新🤞
    ————————————————-———————————————-————————

题目描述

给定一个排序数组和一个目标值,在数组中找到目标值,如果找不到则返回可以将其插入的位置以保证数组仍然有序。

你可以假设数组中无重复元素。

示例 1:

输入: [1,3,5,6], 5
输出: 2

示例 2:

输入: [1,3,5,6], 2
输出: 1

示例 3:

输入: [1,3,5,6], 7
输出: 4

示例 4:

输入: [1,3,5,6], 0
输出: 0

原题:力扣 35. 搜索插入位置

思路及实现

方式一:二分查找

思路

题目要求在一个有序数组中查找目标值,如果找不到则返回可以将其插入的位置以保证数组仍然有序。由于数组是有序的,所以我们可以使用二分查找算法来优化搜索过程。

二分查找的基本思路是,每次取数组的中间元素与目标值进行比较:

  • 如果中间元素正好是要查找的目标值,则搜索结束;
  • 如果目标值大于或小于中间元素,则在数组大于或小于中间元素的那一半中查找,而且跟开始一样从中间元素开始比较。

在二分查找的过程中,我们可以同时记录可以插入目标值的位置。如果目标值大于中间元素,说明目标值应该插入在右半部分的起始位置,这个位置正好是中间元素的下一个位置;如果目标值小于中间元素,说明目标值应该插入在左半部分的末尾位置,这个位置正好是中间元素的位置。

代码实现

Java版本
public class Solution {
    public int searchInsert(int[] nums, int target) {
        int left = 0, right = nums.length - 1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] == target) {
                return mid;
            } else if (nums[mid] < target) {
                left = mid + 1;
            } else {
                right = mid - 1;
            }
        }
        // 退出循环时,left > right,left 的位置就是可以插入 target 的位置
        return left;
    }
}

说明:
Java版本的实现中,我们定义了两个指针leftright,分别表示数组的起始位置和结束位置。在while循环中,我们计算中间位置mid,并根据nums[mid]与目标值target的比较结果来更新leftright的值。最终,当循环结束时,left的值就是可以插入target的位置。

C语言版本
int searchInsert(int* nums, int numsSize, int target) {
    int left = 0, right = numsSize - 1;
    while (left <= right) {
        int mid = left + (right - left) / 2;
        if (nums[mid] == target) {
            return mid;
        } else if (nums[mid] < target) {
            left = mid + 1;
        } else {
            right = mid - 1;
        }
    }
    return left;
}

说明:
C语言版本的实现与Java版本类似,但是需要注意在C语言中,数组的大小需要作为函数的参数传递。

Python3版本
class Solution:
    def searchInsert(self, nums: List[int], target: int) -> int:
        left, right = 0, len(nums) - 1
        while left <= right:
            mid = (left + right) // 2
            if nums[mid] == target:
                return mid
            elif nums[mid] < target:
                left = mid + 1
            else:
                right = mid - 1
        return left

说明:
Python3版本的实现也采用了二分查找的思路,并且使用了整数除法//来避免浮点数。

Golang版本
package main

import "fmt"

func searchInsert(nums []int, target int) int {
    left, right := 0, len(nums)-1
    for left <= right {
        mid := left + (right-left)/2
        if nums[mid] ==target {
            return mid
        } else if nums[mid] < target {
            left = mid + 1
        } else {
            right = mid - 1
        }
    }
    return left
}

func main() {
    nums := []int{1, 3, 5, 6}
    target := 5
    result := searchInsert(nums, target)
    fmt.Println(result) // 输出: 2
}

说明:
Golang版本的实现与前面几种语言类似,同样使用了二分查找算法来寻找目标值或者插入位置。

复杂度分析

  • 时间复杂度:O(log n),其中 n 是数组的长度。二分查找每次都将搜索范围减半,因此时间复杂度是对数级别的。

  • 空间复杂度:O(1)。我们只使用了常量级别的额外空间来存储指针和中间变量。

方式二:线性搜索

思路

虽然题目中给出了数组是有序的,但我们也可以使用线性搜索(即遍历数组)的方式来解决问题。对于每个数组元素,我们比较它是否等于目标值,或者是否小于目标值以确定插入位置。

代码实现

Java版本
public class Solution {
    public int searchInsert(int[] nums, int target) {
        for (int i = 0; i < nums.length; i++) {
            if (nums[i] >= target) {
                return i;
            }
        }
        // 如果遍历完整个数组都没有找到目标值,说明目标值应该插入在数组末尾
        return nums.length;
    }
}

说明:
Java版本的实现中,我们遍历数组,一旦找到某个元素大于等于目标值,就返回当前位置。如果遍历完整个数组都没有找到,则返回数组长度,表示目标值应该插入在数组末尾。

C语言版本
int searchInsert(int* nums, int numsSize, int target) {
    for (int i = 0; i < numsSize; i++) {
        if (nums[i] >= target) {
            return i;
        }
    }
    return numsSize;
}

说明:
C语言版本的实现与Java版本类似,但是需要注意在C语言中,数组的大小是作为函数的参数传递的。

Python3版本
class Solution:
    def searchInsert(self, nums: List[int], target: int) -> int:
        for i in range(len(nums)):
            if nums[i] >= target:
                return i
        return len(nums)

说明:
Python3版本使用for循环遍历数组,一旦找到大于等于目标值的元素,就返回其索引。

Golang版本
package main

import "fmt"

func searchInsert(nums []int, target int) int {
    for i, num := range nums {
        if num >= target {
            return i
        }
    }
    return len(nums)
}

func main() {
    nums := []int{1, 3, 5, 6}
    target := 5
    result := searchInsert(nums, target)
    fmt.Println(result) // 输出: 2
}

说明:
Golang版本的实现使用range关键字遍历数组,与Python3版本类似。

复杂度分析

  • 时间复杂度:O(n),其中 n 是数组的长度。在最坏情况下,我们需要遍历整个数组才能找到插入位置。

  • 空间复杂度:O(1)。我们同样只使用了常量级别的额外空间来存储索引和中间变量。

总结

方式优点缺点时间复杂度空间复杂度
方式一(二分查找)效率高,时间复杂度低需要数组有序O(log n)O(1)
方式二(线性搜索)代码简单,容易理解时间复杂度较高O(n)O(1)

相似题目

相似题目难度链接
leetcode 34. 在排序数组中查找元素的第一个和最后一个位置中等力扣-34

欢迎一键三连(关注+点赞+收藏),技术的路上一起加油!!!代码改变世界

  • 关于我:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名),回复暗号,更能获取学习秘籍和书籍等

  • —⬇️欢迎关注下面的公众号:进朱者赤,认识不一样的技术人。⬇️—

  • 73
    点赞
  • 49
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

进朱者赤

多多支持

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值