【leetcode】16 3Sum Closest

题目说明

https://leetcode-cn.com/problems/3sum-closest/description/
给定一个包括 n 个整数的数组 nums 和 一个目标值 target。找出 nums 中的三个整数,使得它们的和与 target 最接近。返回这三个数的和。假定每组输入只存在唯一答案。

解法

与3sum方法类似,需要进行两层循环的遍历。
首先对数组进行排序,对于有序数组,就可以采用对撞指针(夹逼)的方法进行遍历了。
第一层作为第一个数进行遍历,第二层采用对撞指针(夹逼)的方法,后两数的和要不断趋近于 target-第一个数。
时间复杂度:O(n^2)

int threeSumClosest(vector<int>& nums, int target) {
    int res = 0;
    //对小于等于三个数的情况作处理
    if (nums.size() <= 3){
        for (int i = 0; i < nums.size(); i ++)
        {
            res += nums[i];
        }
        return res;
    }
    //从小到大排序
    sort(nums.begin(), nums.end());
    //初始值设置为前三个之和,应该是最小值
    int min = abs(target - nums[0] -nums[1] - nums[2]);
    res = nums[0] + nums[1] + nums[2];
    for (int i = 0; i < nums.size() - 2; i ++)
    {
        //获取差离目标的差值
        int sum = target - nums[i];
        int l = i + 1;
        int r = nums.size() - 1;
        while (l < r)
        {
            if (nums[l] + nums[r] == sum)
                return target;
            else{
                //计算间隔
                int tmpmin = abs(sum - nums[l] - nums[r]);
                //若小于原间隔,则记录最小间隔值
                if (min > tmpmin){
                    min = tmpmin;
                    res = nums[i] + nums[l] + nums[r];
                }
                //若后两数之和大于差值,则需要减小,右边界左移
                if(nums[l] + nums[r] > sum) {
                    r --;
                    while(l < r && nums[r] == nums[r+1])//去重
                        r--;
                }else{//若后两数之和小于差值,则需要增加,左边界右移
                    l ++;
                    while(l < r && nums[l] == nums[l-1])//去重
                        l++;
                }
            }
        }

        while(i < nums.size() - 2 && nums[i] == nums[i+1])//去重
            i++;
    }
    return res;
}
posted @ 2018-08-22 22:39  JESSET  阅读(162)  评论(0编辑  收藏  举报