(新A卷,100分)- 两数之和绝对值最小(Java & JS & Python)华为OD

(新A卷,100分)- 两数之和绝对值最小(Java & JS & Python)华为OD

题目描述

给定一个从小到大的有序整数序列(存在正整数和负整数)数组 nums ,请你在该数组中找出两个数,其和的绝对值(|nums[x]+nums[y]|)为最小值,并返回这个绝对值。
每种输入只会对应一个答案。但是,数组中同一个元素不能使用两遍。

输入描述

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

输出描述

两数之和绝对值最小值

用例
输入-3 -1 5 7 11 15
输出2
说明因为 |nums[0] + nums[2]| = |-3 + 5| = 2 最小,所以返回 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 int getResult(int[] nums) {
    int min = Integer.MAX_VALUE;
 
    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]);
		min = Math.min(min, sum);
      }
    }
 
    return min;
  }
}

二分查找优化

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 int 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];

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

    // 下面是有正有负的处理逻辑
    int min = Integer.MAX_VALUE;

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

    // 非负数部分的前两个值
    if (idx < n - 1) {
      min = Math.min(min, Math.abs(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;

      min = Math.min(min, Math.abs(nums[i] + positive[j]));

      if (j - 1 >= 0) {
        min = Math.min(min, Math.abs(nums[i] + positive[j - 1]));
      }
    }

    return min;
  }
}
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;

  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]);
	  min = Math.min(min, sum);
    }
  }

  return min;
}

二分查找优化

/* 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];

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

  // 下面是有正有负的处理逻辑
  let min = Infinity;

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

  // 非负数部分的前两个值
  if (idx < n - 1) {
	  min = Math.min(min, Math.abs(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;

	min = Math.min(min, Math.abs(nums[i] + positive[j]));

    if (j - 1 >= 0) {
	  min = Math.min(min, Math.abs(nums[i] + positive[j - 1]));
    }
  }

  return min;
}

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 getResult():
    minV = sys.maxsize

    for i in range(len(nums)):
        for j in range(i + 1, len(nums)):
            sumV = abs(nums[i] + nums[j])
            minV = min(minV, sumV)

    return minV


# 算法调用
print(getResult())

二分查找优化

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 getResult():
    # nums.sort()

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

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

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

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

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

    # 非负数部分的前两个值
    if idx < n - 1:
        minV = min(minV, abs(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

        minV = min(minV, abs(nums[i] + positive[j]))

        if j - 1 >= 0:
            minV = min(minV, abs(nums[i] + positive[j - 1]))

    return minV


# 算法调用
print(getResult())
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值