(B卷,100分)- 乱序整数序列两数之和绝对值最小(Java & JS & Python)

题目描述

给定一个随机的整数(可能存在正整数和负整数)数组 nums,请你在该数组中找出两个数,其和的绝对值(|nums[x]+nums[y]|)为最小值,并返回这个两个数(按从小到大返回)以及绝对值。

每种输入只会对应一个答案。但是,数组中同一个元素不能使用两遍。

输入描述

一个通过空格分割的有序整数序列字符串,最多1000个整数,且整数数值范围是 [-65535, 65535]。

输出描述

两数之和绝对值最小值

用例

输入 -1 -3 7 5 11 15
输出 -3 5 2
说明 因为 |nums[0] + nums[2]| = |-3 + 5| = 2 最小,所以返回 -3 5 2。

题目解析

本题数量级不大,可以考虑暴力破解。

Java算法源码

import java.util.Arrays;
import java.util.Scanner;

public class Main {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int[] nums = Arrays.stream(sc.nextLine().split(" ")).mapToInt(Integer::parseInt).toArray();
    System.out.println(getResult(nums));
  }

  public static String getResult(int[] nums) {
    int min = Integer.MAX_VALUE;
    String ans = "";

    for (int i = 0; i < nums.length; i++) {
      for (int j = i + 1; j < nums.length; j++) {
        int sum = Math.abs(nums[i] + nums[j]);
        if (min > sum) {
          min = sum;
          ans = (nums[i] < nums[j] ? nums[i] + " " + nums[j] : nums[j] + " " + nums[i]) + " " + sum;
        }
      }
    }

    return ans;
  }
}

JS算法源码

/* JavaScript Node ACM模式 控制台输入获取 */
const readline = require("readline");

const rl = readline.createInterface({
  input: process.stdin,
  output: process.stdout,
});

rl.on("line", (line) => {
  const nums = line.split(" ").map(Number);
  console.log(getResult(nums));
});

function getResult(nums) {
  let min = Infinity;
  let ans = "";

  for (let i = 0; i < nums.length; i++) {
    for (let j = i + 1; j < nums.length; j++) {
      const sum = Math.abs(nums[i] + nums[j]);
      if (min > sum) {
        min = sum;
        ans =
          nums[i] < nums[j]
            ? `${nums[i]} ${nums[j]} ${sum}`
            : `${nums[j]} ${nums[i]} ${sum}`;
      }
    }
  }

  return ans;
}

Python算法源码

# 输入获取
import sys

nums = list(map(int, input().split()))


# 算法入口
def getResult():
    minV = sys.maxsize
    ans = ""

    for i in range(len(nums)):
        for j in range(i + 1, len(nums)):
            sumV = abs(nums[i] + nums[j])
            if minV > sumV:
                minV = sumV
                ans = f"{nums[i]} {nums[j]} {sumV}" if nums[i] < nums[j] else f"{nums[j]} {nums[i]} {sumV}"

    return ans


# 算法调用
print(getResult())

优化解法

本题的优化思路是:

首先将nums数组升序,nums数组长度为n

  • 若nums数组全是非负数,比如nums = [0,1,2,3,4],则绝对值最小和 = nums[0] + nums[1]
  • 若nums数组全是负数,比如nums = [-4,-3,-2,-1],则绝对值最小和 = nums[n-2] + nums[n-1]
  • 若nums数组有非负数,也有负数,假设第一个非负数位置是idx,那么此时绝对值最小和可能是:
  1. 非负数部分的前两个值:nums[idx] + nums[idx+1]
  2. 负数部分最后两个值:nums[idx-2] + nums[idx-1]
  3. 非负positive + 负negative的组合,组合中 -negative和positive必然是最接近,此时才能保证组合的和的绝对值最小

找第一个非负数位置是idx,我们可以使用二分查找目标值0在nums中的有序插入位置

  • 如果idx == 0,那么说明nums中全是非负数
  • 如果idx == n,那么说明nums中全是负数
  • 如果0 < idx < n,那么说明nums中既又负数,也有非负数,其中[0, idx-1]是负数部分,[idx, n-1]是非负数部分

Java算法源码

import java.util.Arrays;
import java.util.Scanner;

public class Main {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int[] nums = Arrays.stream(sc.nextLine().split(" ")).mapToInt(Integer::parseInt).toArray();
    System.out.println(getResult(nums));
  }

  public static String getResult(int[] nums) {
    Arrays.sort(nums);

    int idx = Arrays.binarySearch(nums, 0);
    if (idx < 0) idx = -idx - 1;

    // 全正数,或者 0+多个正数
    if (idx == 0) return nums[0] + " " + nums[1] + " " + (nums[0] + nums[1]);

    int n = nums.length;
    // 全负数,或者 多个负数+0
    if (idx >= n - 1)
      return nums[n - 2] + " " + nums[n - 1] + " " + Math.abs(nums[n - 2] + nums[n - 1]);

    // 下面是有正有负的处理逻辑
    int[] min = {Integer.MAX_VALUE};
    String[] ans = {""};

    // 负数部分最后两个值
    if (idx >= 2) {
      check(min, ans, nums[idx - 2], nums[idx - 1]);
    }

    // 非负数部分的前两个值
    if (idx < n - 1) {
      check(min, ans, nums[idx], nums[idx + 1]);
    }

    // 非负数部分的数组
    int[] positive = Arrays.copyOfRange(nums, idx, n);
    for (int i = 0; i < idx; i++) {
      // 注意通过二分查找-nums[i]在positive的有序插入位置j,则最接近-nums[i]的值的位置有两个:j-1和j,其中j位置的元素值 >= -nums[i],而j -
      // 1位置的元素值 < -nums[i]
      int j = Arrays.binarySearch(positive, -nums[i]);

      if (j < 0) j = -j - 1;
      if (j == positive.length) j -= 1;

      check(min, ans, nums[i], positive[j]);

      if (j - 1 >= 0) {
        check(min, ans, nums[i], positive[j - 1]);
      }
    }

    return ans[0];
  }

  public static void check(int[] min, String[] ans, int num1, int num2) {
    int sum = Math.abs(num1 + num2);
    if (min[0] > sum) {
      min[0] = sum;
      ans[0] = num1 + " " + num2 + " " + sum;
    }
  }
}

JS算法源码

/* JavaScript Node ACM模式 控制台输入获取 */
const readline = require("readline");

const rl = readline.createInterface({
  input: process.stdin,
  output: process.stdout,
});

rl.on("line", (line) => {
  const nums = line.split(" ").map(Number);
  console.log(getResult(nums));
});

function getResult(nums) {
  nums.sort((a, b) => a - b);

  let idx = binarySearch(nums, 0);
  if (idx < 0) idx = -idx - 1;

  // 全正数,或者 0+多个正数
  if (idx == 0) return nums[0] + " " + nums[1] + " " + (nums[0] + nums[1]);

  const n = nums.length;
  // 全负数,或者 多个负数+0
  if (idx >= n - 1)
    return (
      nums[n - 2] +
      " " +
      nums[n - 1] +
      " " +
      Math.abs(nums[n - 2] + nums[n - 1])
    );

  // 下面是有正有负的处理逻辑
  const min = [Infinity];
  const ans = [""];

  // 负数部分最后两个值
  if (idx >= 2) check(min, ans, nums[idx - 2], nums[idx - 1]);

  // 非负数部分的前两个值
  if (idx < n - 1) check(min, ans, nums[idx], nums[idx + 1]);

  // 非负数部分的数组
  const positive = nums.slice(idx);
  for (let i = 0; i < idx; i++) {
    // 注意通过二分查找-nums[i]在positive的有序插入位置j,则最接近-nums[i]的值的位置有两个:j-1和j,其中j位置的元素值 >= -nums[i],而j -
    // 1位置的元素值 < -nums[i]
    let j = binarySearch(positive, -nums[i]);

    if (j < 0) j = -j - 1;
    if (j == positive.length) j -= 1;

    check(min, ans, nums[i], positive[j]);

    if (j - 1 >= 0) {
      check(min, ans, nums[i], positive[j - 1]);
    }
  }

  return ans[0];
}

function check(min, ans, num1, num2) {
  const sum = Math.abs(num1 + num2);
  if (min[0] > sum) {
    min[0] = sum;
    ans[0] = `${num1} ${num2} ${sum}`;
  }
}

function binarySearch(arr, target) {
  let low = 0;
  let high = arr.length - 1;

  while (low <= high) {
    const mid = (low + high) >> 1;
    const midVal = arr[mid];

    if (midVal > target) {
      high = mid - 1;
    } else if (midVal < target) {
      low = mid + 1;
    } else {
      return mid;
    }
  }

  return -low - 1;
}

Python算法源码

import sys

# 输入获取
nums = list(map(int, input().split()))


def binarySearch(arr, target):
    low = 0
    high = len(arr) - 1

    while low <= high:
        mid = (low + high) >> 1
        midVal = arr[mid]

        if midVal > target:
            high = mid - 1
        elif midVal < target:
            low = mid + 1
        else:
            return mid

    return -low - 1


def check(minV, ans, num1, num2):
    sumV = abs(num1 + num2)
    if minV[0] > sumV:
        minV[0] = sumV
        ans[0] = f"{num1} {num2} {sumV}"


# 算法入口
def getResult():
    nums.sort()

    idx = binarySearch(nums, 0)
    if idx < 0:
        idx = -idx - 1

    # 全正数,或者 0+多个正数
    if idx == 0:
        return f"  {nums[0] + nums[1]}"

    n = len(nums)
    # 全负数,或者 多个负数+0
    if idx >= n-1:
        return f"{nums[n-2]} {nums[n-1]} {abs(nums[n-2] + nums[n-1])}"

    # 下面是有正有负的处理逻辑
    minV = [sys.maxsize]
    ans = [""]

    # 负数部分最后两个值
    if idx >= 2:
        check(minV, ans, nums[idx-2], nums[idx-1])

    # 非负数部分的前两个值
    if idx < n-1:
        check(minV, ans, nums[idx], nums[idx+1])

    # 非负数部分的数组
    positive = nums[idx:]
    for i in range(0, idx):
        # 注意通过二分查找-nums[i]在positive的有序插入位置j,则最接近-nums[i]的值的位置有两个:j-1和j,其中j位置的元素值 >= -nums[i],而j - 1位置的元素值 < -nums[i]
        j = binarySearch(positive, -nums[i])

        if j < 0:
            j = -j - 1

        if j == len(positive):
            j -= 1

        check(minV, ans, nums[i], positive[j])

        if j - 1 >= 0:
            check(minV, ans, nums[i], positive[j-1])

    return ans[0]


# 算法调用
print(getResult())

免责声明:

1、IT资源小站为非营利性网站,全站所有资料仅供网友个人学习使用,禁止商用
2、本站所有文档、视频、书籍等资料均由网友分享,本站只负责收集不承担任何技术及版权问题
3、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除下载链接并致以最深的歉意
4、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责
5、一经注册为本站会员,一律视为同意网站规定,本站管理员及版主有权禁止违规用户
6、其他单位或个人使用、转载或引用本文时必须同时征得该帖子作者和IT资源小站的同意
7、IT资源小站管理员和版主有权不事先通知发贴者而删除本文

0

评论0

站点公告

没有账号?注册  忘记密码?