给定一个整数数组nums
,请你在该数组中找出两个数,使得这两个数的和的绝对值abs(nums[x] + nums[y])
为最小值并按从小到大返回这两个数以及它们和的绝对值。每种输入只会对应一个答案。数组中同一个元素不能使用两遍。
一个通过空格分割的整数序列字符串,最多1000
个整数,且整数数值范围是[-65535,65535]
两个数以及两数之和绝对值
-1 -3 7 5 11 15
-3 5 2
因为abs(nums[0]+nums[2]) = abs(-3+5) = 2
在所有数对中最小,所以返回-3 5 2
注意,本题和LC167. 两数之和 II - 输入有序数组解题方法非常类似。但在判断指针移动的条件上,稍微有些不同。
要用上相向双指针,首先要对lst
数组进行排序。
两个数的和的绝对值越小,说明这两个数的和越接近于0
。所以这道题可以换一种问法,找到两个数使得它们的和尽可能地接近0
。我们可以使用相向双指针left
和right
指向一小一大两个数,当
lst[left] + lst[right] > 0
时,如果令left
右移,两数和会进一步增大,距离0
更远,因此需要令right
左移。lst[left] + lst[right] < 0
时,如果令right
左移,两数和会进一步减小,距离0
更远,因此需要令left
右移。lst[left] + lst[right] == 0
时,两数和的绝对值已经达到最小,可以直接退出循环。另外,对于每对两数和sum2 = lst[left] + lst[right]
,我们都应该令其绝对值与之前的得到最小两数和绝对值进行比较并更新。上述思路整理为代码即:
while left < right:
sum2 = lst[left] + lst[right]
if abs(sum2) < ans[2]:
ans = [lst[left], lst[right], abs(sum2)]
if sum2 > 0:
right -= 1
elif sum2 < 0:
left += 1
else:
break
如果没有想到上述方法,在选择left
和right
移动哪一个的问题上,我们也可以直接考虑移动后的结果。当
abs(lst[left+1] + lst[right]) < abs(lst[left] + lst[right-1])
,即left
右移后两数和的绝对值小于right
左移的结果,那么我们令left
右移。abs(lst[left+1] + lst[right]) >= abs(lst[left] + lst[right-1])
,即left
右移后两数和的绝对值大于等于right
左移的结果,那么我们令right
左移。上述思路整理为代码即:
while left < right:
if abs(lst[left] + lst[right]) < ans[2]:
ans = [lst[left], lst[right], abs(sum2)]
if abs(lst[left+1] + lst[right]) < abs(lst[left] + lst[right-1]):
left += 1
else:
right -= 1
将题目转化为考虑最接近target = 0
的两数和
# 题目:2023Q1A-两数之和绝对值最小
# 分值:100
# 作者:许老师-闭着眼睛学数理化
# 算法:相向双指针
# 代码看不懂的地方,请直接在群上提问
lst = list(map(int, input().split()))
lst.sort()
n = len(lst)
# 答案变量ans为一个三元列表,分别为两个数以及两数和的绝对值
ans = [lst[0], lst[-1], abs(lst[0]+lst[-1])]
left, right = 0, n-1
while left < right:
# 计算left和right所指两个元素的和
sum2 = lst[left] + lst[right]
# 如果两数和的绝对值小于之前记录的最小两数和的绝对值
if abs(sum2) < ans[2]:
# 更新答案变量ans
ans = [lst[left], lst[right], abs(sum2)]
# 两数和大于0,right左移
if sum2 > 0:
right -= 1
# 两数和小于0,left右移
elif sum2 < 0:
left += 1
# 两数和等于0,退出循环
else:
break
# 用map把ans中的所有int转为str,合并后输出
print(" ".join(map(str, ans)))
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String input = scanner.nextLine();
List<Integer> lst = new ArrayList<>();
while (input.contains(" ")) {
int pos = input.indexOf(' ');
lst.add(Integer.parseInt(input.substring(0, pos)));
input = input.substring(pos + 1);
}
lst.add(Integer.parseInt(input));
lst.sort(null);
int n = lst.size();
List<Integer> ans = Arrays.asList(lst.get(0), lst.get(n - 1), Math.abs(lst.get(0) + lst.get(n - 1)));
int left = 0;
int right = n - 1;
while (left < right) {
int sum2 = lst.get(left) + lst.get(right);
if (Math.abs(sum2) < ans.get(2)) {
ans = Arrays.asList(lst.get(left), lst.get(right), Math.abs(sum2));
}
if (sum2 > 0) {
right--;
} else if (sum2 < 0) {
left++;
} else {
break;
}
}
for (int num : ans) {
System.out.print(num + " ");
}
System.out.println();
}
}
#include
#include
#include
using namespace std;
int main() {
vector<int> lst;
string input;
getline(cin, input);
size_t pos = 0;
while ((pos = input.find(' ')) != string::npos) {
lst.push_back(stoi(input.substr(0, pos)));
input.erase(0, pos + 1);
}
lst.push_back(stoi(input));
sort(lst.begin(), lst.end());
int n = lst.size();
vector<int> ans = {lst[0], lst[n - 1], abs(lst[0] + lst[n - 1])};
int left = 0;
int right = n - 1;
while (left < right) {
int sum2 = lst[left] + lst[right];
if (abs(sum2) < ans[2]) {
ans = {lst[left], lst[right], abs(sum2)};
}
if (sum2 > 0) {
right--;
} else if (sum2 < 0) {
left++;
} else {
break;
}
}
for (int num : ans) {
cout << num << " ";
}
cout << endl;
return 0;
}
直接考虑移动哪一个指针,能使两数和的绝对值更小。
# 题目:2023Q1A-两数之和绝对值最小
# 分值:100
# 作者:许老师-闭着眼睛学数理化
# 算法:相向双指针
# 代码看不懂的地方,请直接在群上提问
lst = list(map(int, input().split()))
lst.sort()
n = len(lst)
# 答案变量ans为一个三元列表,分别为两个数以及两数和的绝对值
ans = [lst[0], lst[-1], abs(lst[0]+lst[-1])]
left, right = 0, n-1
while left < right:
# 如果当前两数和绝对值小于当前记录的最小值,则更新答案
if abs(lst[left] + lst[right]) < ans[2]:
ans = [lst[left], lst[right], abs(sum2)]
# left和right移动哪个能使得两数和绝对值更小,则移动哪一个
if abs(lst[left+1] + lst[right]) < abs(lst[left] + lst[right-1]):
left += 1
else:
right -= 1
# 用map把ans中的所有int转为str,合并后输出
print(" ".join(map(str, ans)))
import java.util.*;
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String input = scanner.nextLine();
String[] tokens = input.split("\\s+");
int[] lst = new int[tokens.length];
for (int i = 0; i < tokens.length; i++) {
lst[i] = Integer.parseInt(tokens[i]);
}
Arrays.sort(lst);
int n = lst.length;
// 答案变量ans为一个三元数组,分别为两个数以及两数和的绝对值
int[] ans = new int[]{lst[0], lst[n - 1], Math.abs(lst[0] + lst[n - 1])};
int left = 0;
int right = n - 1;
while (left < right) {
// 如果当前两数和绝对值小于当前记录的最小值,则更新答案
if (Math.abs(lst[left] + lst[right]) < ans[2]) {
ans = new int[]{lst[left], lst[right], Math.abs(lst[left] + lst[right])};
}
// left和right移动哪个能使得两数和绝对值更小,则移动哪一个
if (Math.abs(lst[left + 1] + lst[right]) < Math.abs(lst[left] + lst[right - 1])) {
left++;
} else {
right--;
}
}
// 输出结果
for (int num : ans) {
System.out.print(num + " ");
}
}
}
#include
#include
#include
#include
using namespace std;
int main() {
string input;
getline(cin, input);
stringstream ss(input);
vector<int> lst;
int num;
while (ss >> num) {
lst.push_back(num);
}
sort(lst.begin(), lst.end());
int n = lst.size();
vector<int> ans = {lst[0], lst[n - 1], abs(lst[0] + lst[n - 1])};
int left = 0;
int right = n - 1;
while (left < right) {
if (abs(lst[left] + lst[right]) < ans[2]) {
ans = {lst[left], lst[right], abs(lst[left] + lst[right])};
}
if (abs(lst[left + 1] + lst[right]) < abs(lst[left] + lst[right - 1])) {
left++;
} else {
right--;
}
}
for (int num : ans) {
cout << num << " ";
}
cout << endl;
return 0;
}
时间复杂度:O(NlogN)
。主要为排序的时间复杂度。
空间复杂度:O(1)
。忽略排序所需要的栈空间,仅需要用到若干常数变量。
华为OD算法/大厂面试高频题算法冲刺训练目前开始常态化报名!目前已服务100+同学成功上岸!
课程讲师为全网50w+粉丝编程博主@吴师兄学算法 以及小红书头部编程博主@闭着眼睛学数理化
每期人数维持在20人内,保证能够最大限度地满足到每一个同学的需求,达到和1v1同样的学习效果!
60+天陪伴式学习,40+直播课时,300+动画图解视频,300+LeetCode经典题,200+华为OD真题/大厂真题,还有简历修改、模拟面试、专属HR对接将为你解锁
可上全网独家的欧弟OJ系统练习华子OD、大厂真题
可查看链接 大厂真题汇总 & OD真题汇总(持续更新)
绿色聊天软件戳 od1336
了解更多