经典京东面试原题

写在前面

今天在「京东」题库中翻到一道经典题。

众所周知,题目越经典,评论区越逆天。

说归说,闹归闹,这道题还是要掌握的。

毕竟除了京东,汇量科技也在测试开发中考过:

下面一起来看看吧 ~

题目描述

平台:LeetCode

题号:16

给定一个包括 n n n 个整数的数组 nums 和 一个目标值 target

找出 nums 中的三个整数,使得它们的和与 target 最接近。

返回这三个数的和。

每组输入只存在唯一答案。

示例:

ini 复制代码
输入:nums = [-1,2,1,-4], target = 1

输出:2

解释:与 target 最接近的和是 2 (-1 + 2 + 1 = 2) 。

提示:

  • 3 < = n u m s . l e n g t h < = 1 0 3 3 <= nums.length <= 10^3 3<=nums.length<=103
  • − 1 0 3 < = n u m s i < = 1 0 3 -10^3 <= numsi <= 10^3 −103 <=numsi <=103
  • − 1 0 4 < = t a r g e t < = 1 0 4 -10^4 <= target <= 10^4 −104 <=target <=104

排序 + 双指针

对数组进行排序,使用三个指针 ijk 分别代表要找的三个数。

  1. 通过枚举 i 确定第一个数,另外两个指针 jk 分别从左边 i + 1 和右边 n - 1 往中间移动,找到满足 nums[i] + nums[j] + nums[k] 最接近 target 的唯一解。

  2. jk 指针的移动逻辑,分情况讨论 sum = nums[i] + nums[j] + nums[k]

    • sum > targetk 左移,使 sum 变小
    • sum < targetj 右移,使 sum 变大
    • sum = target:找到最符合要求的答案,直接返回

为了更快找到答案,对于相同的 i,可以直接跳过下标。

Java 代码:

Java 复制代码
class Solution {
    public int threeSumClosest(int[] nums, int target) {
        Arrays.sort(nums);
        int n = nums.length, ans = nums[0] + nums[1] + nums[2];
        for (int i = 0; i < n; i++) {
            if (i > 0 && nums[i] == nums[i - 1]) continue;
            int j = i + 1, k = n - 1;
            while (j < k) {
                int sum = nums[i] + nums[j] + nums[k];
                if (Math.abs(sum - target) < Math.abs(ans - target)) ans = sum;
                if (ans == target) {
                    return target;
                } else if (sum > target) {
                    k--;
                } else if (sum < target) {
                    j++;
                }
            }
        }
        return ans;
    }
}

C++ 代码:

C++ 复制代码
class Solution {
public:
    int threeSumClosest(vector<int>& nums, int target) {
        sort(nums.begin(), nums.end());
        int n = nums.size(), ans = nums[0] + nums[1] + nums[2];
        for (int i = 0; i < n; i++) {
            if (i > 0 && nums[i] == nums[i - 1]) continue;
            int j = i + 1, k = n - 1;
            while (j < k) {
                int sum = nums[i] + nums[j] + nums[k];
                if (abs(sum - target) < abs(ans - target)) ans = sum;
                if (ans == target) {
                    return target;
                } else if (sum > target) {
                    k--;
                } else if (sum < target) {
                    j++;
                }
            }
        }
        return ans;
    }
};

Python 代码:

Python 复制代码
class Solution:
    def threeSumClosest(self, nums: List[int], target: int) -> int:
        nums.sort()
        n, ans = len(nums), nums[0] + nums[1] + nums[2]
        for i in range(n):
            if i > 0 and nums[i] == nums[i - 1]: 
                continue
            j, k = i + 1, n - 1
            while j < k:
                s = nums[i] + nums[j] + nums[k]
                if abs(s - target) < abs(ans - target):
                    ans = s
                if ans == target:
                    return target
                elif s > target:
                    k -= 1
                elif s < target:
                    j += 1
        return ans

TypeScript 代码:

TypeScript 复制代码
function threeSumClosest(nums: number[], target: number): number {
    nums.sort((a, b) => a - b);
    let n = nums.length, ans = nums[0] + nums[1] + nums[2];
    for (let i = 0; i < n; i++) {
        if (i > 0 && nums[i] == nums[i - 1]) continue;
        let j = i + 1, k = n - 1;
        while (j < k) {
            const sum = nums[i] + nums[j] + nums[k];
            if (Math.abs(sum - target) < Math.abs(ans - target)) ans = sum;
            if (ans == target) {
                return target;
            } else if (sum > target) {
                k--;
            } else if (sum < target) {
                j++;
            }
        }
    }
    return ans;
};
  • 时间复杂度:排序的复杂度为 O ( log ⁡ n ) O(\log{n}) O(logn),对于每个 i 而言,最坏的情况 jk 都要扫描一遍数组的剩余部分,复杂度为 O ( n 2 ) O(n^2) O(n2)。整体复杂度为 O ( n 2 ) O(n^2) O(n2)
  • 空间复杂度: O ( n 2 ) O(n ^ 2) O(n2)

更多更全更热门的「笔试/面试」相关资料可访问排版精美的 合集新基地 🎉🎉

相关推荐
Bigger几秒前
GitLab-Runner + AI 代码审查服务 + 远程大模型 全套部署运维实战
前端·ci/cd·ai编程
逍遥德几秒前
PostgreSQL ---【序列】用法详解
数据库·后端·sql·postgresql
逍遥德1 分钟前
PostgreSQL --- 自增主键【序列】的避坑指南
数据库·后端·sql·mysql·postgresql·sqlserver
_xaboy1 分钟前
开源AI表单设计器 FcDesigner v3.5 版本发布!
前端·vue.js·低代码·开源·表单
爱讲故事的2 分钟前
操作系统第三讲:Context Switch —— 用户态如何安全地进入内核态?
前端·javascript·安全
段ヤシ.8 分钟前
【Java框架】知识点汇总Day7:Spring Boot +Vue(持续更新)
vue.js·spring boot·后端·框架
土狗TuGou8 分钟前
SQL进阶笔记 · 第1篇:存储引擎
java·数据库·笔记·后端·sql·mysql
light blue bird12 分钟前
支轴事件任务线程执行工序路径的图表组件
前端·jvm·windows
终端行者13 分钟前
企业级 Jenkins Pipeline 实战Docker构建前端+Ansible发布
前端·ci/cd·docker·jenkins
码语智行16 分钟前
Spring Security自定义AuthenticationManager实现手机号/密码双认证
java·后端·spring