LeetCode167 TwoSumⅡ

LeetCode167 TwoSumⅡ

题目描述

给定一个已按照升序排列 的有序数组,找到两个数使得它们相加之和等于目标数。

函数应该返回这两个下标值 index1 和 index2,其中 index1 必须小于 index2。

说明:

  • 返回的下标值(index1 和 index2)不是从零开始的。
  • 你可以假设每个输入只对应唯一的答案,而且你不可以重复使用相同的元素。

示例1:

输入: numbers = [2, 7, 11, 15], target = 9
输出: [1,2]
解释: 2 与 7 之和等于目标数 9 。因此 index1 = 1, index2 = 2 。

解答

解法一:暴力解法

两遍遍历,即是两层for循环。但是在leetcode上,这么做会超时。时间负责度为O(n2)。

//解法一:暴力求解
public static int[] solve1(int[] nums, int target){
    int[] result = new int[2];
    a: for (int i=0,len = nums.length;i<len;i++){
        for (int j=1;j<len;j++){
            if ( nums[i]+nums[j]==target && i!=j){
                result[0]=i+1;
                result[1]=j+1;
                break a;
            }
        }
    }
    return result;
}
解法二:二分搜索

暴力解法时,我们并没有用到题目给的条件,升序数组。于是我们可以考虑使用二分法。遍历i,对于每一个nums[i],用二分查找法(复杂度为logn)去找target-nums[i]。所以总的时间复杂度为 nlogn。

public static int[] solve2(int[] nums, int target){
    int[] result = new int[2];
    for (int i=0,len=nums.length;i<len;i++){
        int index2 = binarySearch(nums,i+1,len-1, target-nums[i]);
        if (index2!=-1){
            result[0]=i+1;
            result[1]=index2+1;
            break;
        }
    }
    return result;
}

public static int binarySearch(int[] nums, int left, int right, int result){
    //从[left,right]查找
    while (left<=right){
        int mid = left+(right-left)/2;
        if (nums[mid]==result)
            return mid;
        else if (nums[mid]<result){
            left = mid+1;
        }else {
            right = mid-1;
        }
    }
    return -1;
}
解法三:对撞指针

因为是有序数组,所以可以使用两个指针对撞,一个i从0开始,一个j从n-1开始,如果说nums[i]+nums[j]小于target的话,说明这两个其中某个值要变大,那就只能i++,来保证nums[i]变大了。而如果nums[i]+nums[j]大于target的话,说明这两个其中某个值要变小,那就只能j–,来保证nums[j]变小了。

public static int[] solve3(int[] nums, int target){
    int i=0, j=nums.length-1;
    while (i<j){
        if (nums[i]+nums[j]==target){
            int[] result = new int[2];
            result[0]=i+1;
            result[1]=j+1;
            return result;
        }else if (nums[i]+nums[j]<target){
            i++;
        }else{
            j--;
        }
    }
    return null;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值