1. 两数之和

题目

给定一个整数数组和一个目标值,找出数组中和为目标值的两个数。
你可以假设每个输入只对应一种答案,且同样的元素不能被重复利用。

示例

给定 nums = [2, 7, 11, 15], target = 9

因为 nums[0] + nums[1] = 2 + 7 = 9
所以返回 [0, 1]

解法

方法1

直接双重循环遍历数组,查找和值为 target 的两个元素
时间复杂度o(n2)

方法2

用Map集合把作为参数的数组存起来,数组中元素的值作为键Key,而对应的元素下标作为值Value。遍历一次数组(找出一个元素),并同时在Map集合中找另一个,Map集合查找的时间复杂度是o(1),这样整体的时间复杂度就降到了 o(n) 线性阶,最后将数组中找到的其中一个元素的下标,和Map集合中另一个元素(键)作为集合 get() 方法的参数得到的值即元素在原数组中的位置存在数组中返回

代码实现

方法1

public int[] twoSum(int[] nums, int target) {
		int[] arr = new int[2];
		for (int i = 0; i < nums.length; i++) {
	        for (int j = i + 1; j < nums.length; j++) {
	        	if (nums[i] + nums[j] == target) {
	            	arr[0] = i;
	            	arr[1] = j;
	            	return arr;
	            }
	        }
	    }
	    throw new IllegalArgumentException("No two sum solution");
	}

注:方法中返回数组的语句可以直接写成:

return new int[] { i, j };

这样将原来的四句代码变为了一句

时间复杂度:o(n2)
空间复杂度:o(1)

方法2

第一段: 先存再找

public int[] twoSum(int[] nums, int target) {
	    Map<Integer, Integer> map = new HashMap<Integer, Integer>();
	    for (int i = 0; i < nums.length; i++) {
	        map.put(nums[i], i);//先遍历一遍数组,将数组全部放入Map集合中
	    }
	    for (int i = 0; i < nums.length; i++) {//开始找了,先遍历数组
	        if (map.containsKey(target - nums[i]) && map.get(target - nums[i]) != i) {
	        //直接在集合中找符合条件的元素
	            return new int[] { i, map.get(target - nums[i]) };
	        }
	    }
	    throw new IllegalArgumentException("No two sum solution");
	}

第二段: 边存边找

public int[] twoSum(int[] nums, int target) {
		Map<Integer, Integer> map = new HashMap<Integer, Integer>();
	    for (int i = 0; i < nums.length; i++) {//只遍历一次数组
	        if (map.containsKey(target - nums[i])) {//在集合中找符合条件的元素
	            return new int[] { map.get(target - nums[i]), i };
	        }
	        map.put(nums[i], i);//将数组中的元素存入集合
	    }
	    throw new IllegalArgumentException("No two sum solution");
	}

时间复杂度:o(n)
空间复杂度:o(n)

对比两种方法,要不浪费空间只好牺牲速度,而要提高效率就会牺牲空间

  • 5
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于给定的一组整数a1, a2, a3, …, an,要使两个数之和的绝对值最小,我们需要寻找两个数的组合。 我们可以使用两个指针,一个在数组的开头,一个在数组的末尾。我们将这两个指针指向的数相加,并将其与当前最小绝对值进行比较。如果和的绝对值更小,我们更新最小绝对值,并保存这两个数。然后,我们根据和的正负情况将指针向内移动。 具体算法如下: 1. 将数组按升序排序。 2. 定义两个指针,一个指向数组的开头,另一个指向数组的末尾。 3. 初始化最小绝对值为无穷大。 4. 循环寻找两个数之和,并更新最小绝对值以及保存这两个数。 5. 根据和的正负情况移动指针。 6. 返回保存的两个数。 以下是一个示例代码: def findMinSum(numbers): numbers.sort() # 将数组升序排序 left = 0 # 左指针 right = len(numbers) - 1 # 右指针 minAbsSum = float('inf') # 初始最小绝对值为无穷大 num1 = 0 # 第一个数 num2 = 0 # 第二个数 while left < right: currentSum = numbers[left] + numbers[right] # 当前两数之和 currentAbsSum = abs(currentSum) # 当前两数之和的绝对值 if currentAbsSum < minAbsSum: # 如果当前绝对值更小 minAbsSum = currentAbsSum # 更新最小绝对值 num1 = numbers[left] # 保存第一个数 num2 = numbers[right] # 保存第二个数 if currentSum < 0: # 如果和为负数,说明两数之和偏小 left += 1 # 左指针向右移动一位 else: # 如果和为正数,说明两数之和偏大 right -= 1 # 右指针向左移动一位 return num1, num2 这样,通过以上算法,我们可以找到一组两数之和的绝对值最小的数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值