CONTENTS
- LeetCode 16. 最接近的三数之和(中等)
- LeetCode 17. 电话号码的字母组合(中等)
- LeetCode 18. 四数之和(中等)
LeetCode 16. 最接近的三数之和(中等)
【题目描述】
给你一个长度为 n
的整数数组 nums
和一个目标值 target
。请你从 nums
中选出三个整数,使它们的和与 target
最接近。
返回这三个数的和。
假定每组输入只存在恰好一个解。
【示例1】
输入:nums = [-1,2,1,-4], target = 1
输出:2
解释:与 target 最接近的和是 2 (-1 + 2 + 1 = 2) 。
【示例2】
输入:nums = [0,0,0], target = 1
输出:0
【提示】
3 ≤ n u m s . l e n g t h ≤ 1000 3\le nums.length\le 1000 3≤nums.length≤1000
− 1000 ≤ n u m s [ i ] ≤ 1000 -1000\le nums[i]\le 1000 −1000≤nums[i]≤1000
− 1 0 4 ≤ t a r g e t ≤ 1 0 4 -10^4\le target\le 10^4 −104≤target≤104
【分析】
和第15题相似,和 target
最接近的数可能是大于等于 target
的最小的数,也可能是小于等于 target
的最大的数。前者和第15题一样,枚举指针 i i i,然后用双指针 j j j 和 k k k 求出大于等于 target
的最小的数。由于我们之前的实现方式是将 k k k 从右向左移动,找到 nums[i] + nums[j] + nums[k] >= target
的最小的 k k k,因此我们可以得出 nums[i] + nums[j] + nums[k - 1] < target
且 k − 1 k-1 k−1 是满足该式的最大的 k k k(需要注意保证 k − 1 ≠ j k-1\ne j k−1=j)。
【代码】
class Solution {
public:int threeSumClosest(vector<int>& nums, int target) {sort(nums.begin(), nums.end());pair<int, int> res(INT_MAX, 0);for (int i = 0; i < nums.size(); i++){for (int j = i + 1, k = nums.size() - 1; j < k; j++){while (j < k - 1 && nums[i] + nums[j] + nums[k - 1] >= target) k--;int s1 = nums[i] + nums[j] + nums[k], s2 = nums[i] + nums[j] + nums[k - 1];res = min(res, make_pair(abs(s1 - target), s1)); // 不一定会大于等于0,因此要用absif (j < k - 1) // 确保j和k-1不是一样的才能使用nums[k-1]res = min(res, make_pair(target - s2, s2)); // 一定小于0}}return res.second;}
};
LeetCode 17. 电话号码的字母组合(中等)
【题目描述】
给定一个仅包含数字 2-9
的字符串,返回所有它能表示的字母组合。答案可以按任意顺序返回。
给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。
【示例1】
输入:digits = "23"
输出:["ad","ae","af","bd","be","bf","cd","ce","cf"]
【示例2】
输入:digits = ""
输出:[]
【示例3】
输入:digits = "2"
输出:["a","b","c"]
【提示】
0 ≤ d i g i t s . l e n g t h ≤ 4 0\le digits.length\le 4 0≤digits.length≤4
digits[i]
是范围 ['2', '9']
的一个数字。
【分析】
直接 DFS 爆搜即可。
【Python代码】
class Solution:def letterCombinations(self, digits: str) -> List[str]:d2str = ['', '', 'abc', 'def', 'ghi', 'jkl', 'mno', 'pqrs', 'tuv', 'wxyz']res = []def dfs(digits: str, u: int, now: str):if u == len(digits):res.append(now)returnfor c in d2str[int(digits[u])]:dfs(digits, u + 1, now + c)dfs(digits, 0, '')return [] if not digits else res
LeetCode 18. 四数之和(中等)
【题目描述】
给你一个由 n
个整数组成的数组 nums
,和一个目标值 target
。请你找出并返回满足下述全部条件且不重复的四元组 [nums[a], nums[b], nums[c], nums[d]]
(若两个四元组元素一一对应,则认为两个四元组重复):
0 <= a, b, c, d < n
a
、b
、c
和d
互不相同nums[a] + nums[b] + nums[c] + nums[d] == target
你可以按任意顺序返回答案 。
【示例1】
输入:nums = [1,0,-1,0,-2,2], target = 0
输出:[[-2,-1,1,2],[-2,0,0,2],[-1,0,0,1]]
【示例2】
输入:nums = [2,2,2,2,2], target = 8
输出:[[2,2,2,2]]
【提示】
1 ≤ n u m s . l e n g t h ≤ 200 1\le nums.length\le 200 1≤nums.length≤200
− 1 0 9 ≤ n u m s [ i ] ≤ 1 0 9 -10^9\le nums[i]\le 10^9 −109≤nums[i]≤109
− 1 0 9 ≤ t a r g e t ≤ 1 0 9 -10^9\le target\le 10^9 −109≤target≤109
【分析】
和第15题一样,暴力枚举四个数时间复杂度为 O ( n 4 ) O(n^4) O(n4),使用双指针算法可以优化成 O ( n 3 ) O(n^3) O(n3)。需要注意本题四个数相加可能会溢出。
【代码】
class Solution {
public:vector<vector<int>> fourSum(vector<int>& nums, int target) {sort(nums.begin(), nums.end());vector<vector<int>> res;for (int i = 0; i < nums.size(); i++)if (i && nums[i] == nums[i - 1]) continue;else for (int j = i + 1; j < nums.size(); j++)if (j > i + 1 && nums[j] == nums[j - 1]) continue;else for (int k = j + 1, u = nums.size() - 1; k < u; k++){if (k > j + 1 && nums[k] == nums[k - 1]) continue;while (k < u - 1 && (long long)nums[i] + nums[j] + nums[k] + nums[u - 1] >= target) u--;if ((long long)nums[i] + nums[j] + nums[k] + nums[u] == target)res.push_back({ nums[i], nums[j], nums[k], nums[u] });}return res;}
};