程序员面试题精选100题(10)-排序数组中和为给定值的两个数字[算法]
题目:输入一个已经按升序排序过的数组和一个数字,在数组中查找两个数,使得它们的和正好是输入的那个数字。要求时间复杂度是O(n)。如果有多对数字的和等于输入的数字,输出任意一对即可。
例如输入数组1、2、4、7、11、15和数字15。由于4+11=15,因此输出4和11。
分析:如果我们不考虑时间复杂度,最简单想法的莫过去先在数组中固定一个数字,再依次判断数组中剩下的n-1个数字与它的和是不是等于输入的数字。可惜这种思路需要的时间复杂度是O(n2)。
我们假设现在随便在数组中找到两个数。如果它们的和等于输入的数字,那太好了,我们找到了要找的两个数字;如果小于输入的数字呢?我们希望两个数字的和再大一点。由于数组已经排好序了,我们是不是可以把较小的数字的往后面移动一个数字?因为排在后面的数字要大一些,那么两个数字的和也要大一些,就有可能等于输入的数字了;同样,当两个数字的和大于输入的数字的时候,我们把较大的数字往前移动,因为排在数组前面的数字要小一些,它们的和就有可能等于输入的数字了。
我们把前面的思路整理一下:最初我们找到数组的第一个数字和最后一个数字。当两个数字的和大于输入的数字时,把较大的数字往前移动;当两个数字的和小于数字时,把较小的数字往后移动;当相等时,打完收工。这样扫描的顺序是从数组的两端向数组的中间扫描。
问题是这样的思路是不是正确的呢?这需要严格的数学证明。感兴趣的读者可以自行证明一下。
参考代码:
///
// Find two numbers with a sum in a sorted array
// Output: ture is found such two numbers, otherwise false
///
bool FindTwoNumbersWithSum
(int data[], // a sorted arrayunsigned int length, // the length of the sorted array int sum, // the sumint& num1, // the first number, outputint& num2 // the second number, output
)
{bool found = false;if(length < 1)return found;int ahead = length - 1;int behind = 0;while(ahead > behind){long long curSum = data[ahead] + data[behind];// if the sum of two numbers is equal to the input// we have found themif(curSum == sum){num1 = data[behind];num2 = data[ahead];found = true;break;}// if the sum of two numbers is greater than the input// decrease the greater numberelse if(curSum > sum)ahead --;// if the sum of two numbers is less than the input// increase the less numberelsebehind ++;}return found;
}
扩展(1):输入一个数组,判断这个数组中是不是存在三个数字i, j, k,满足i+j+k等于0。在我的英文博客http://codercareer.blogspot.com/2011/10/no-09-numbers-with-given-sum.html里详细讨论了这个题目。
扩展(2):如果输入的数组是没有排序的,但知道里面数字的范围,其他条件不变,如何在O(n)时间里找到这两个数字?这个的基本思路是先用哈希表实现O(n)的排序(请参照本面试题系列的第57题),接下来的步骤都一样了。
本文已经收录到《剑指Offer——名企面试官精讲典型编程题》一书中,有改动,书中的分析讲解更加详细。欢迎关注。
本题已被九度Online Judge系统收录,欢迎读者移步到http://ac.jobdu.com/hhtproblems.php在线测试自己的代码。
博主何海涛对本博客文章享有版权。网络转载请注明出处http://zhedahht.blog.163.com/。整理出版物请和作者联系。
程序员面试题精选100题(10)-排序数组中和为给定值的两个数字[算法]相关推荐
- 程序员面试题精选100题(41)-把数组排成最小的数[算法]
题目:输入一个正整数数组,将它们连接起来排成一个数,输出能排出的所有数字中最小的一个.例如输入数组{32, 321},则输出这两个能排成的最小数字32132.请给出解决问题的算法,并证明该算法. 分 ...
- 程序员面试题精选100题(42)-旋转数组的最小元素[算法]
题目:把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转.输入一个排好序的数组的一个旋转,输出旋转数组的最小元素.例如数组{3, 4, 5, 1, 2}为{1, 2, 3, 4, 5}的 ...
- 程序员面试题精选100题(29)-调整数组顺序使奇数位于偶数前面[算法]
题目:输入一个整数数组,调整数组中数字的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分.要求时间复杂度为O(n). 分析:如果不考虑时间复杂度,最简单的思路应该是从头扫描这个数组,每 ...
- 程序员面试题精选100题
程序员面试题精选100题(01)-把二元查找树转变成排序的双向链表 题目:输入一棵二元查找树,将该二元查找树转换成一个排序的双向链表.要求不能创建任何新的结点,只调整指针的指向. 比如将二元查找树 ...
- 程序员面试题精选100题:41-50解题报告
程序员面试题精选100题(41)-把数组排成最小的数[算法] 题目:输入一个正整数数组,将它们连接起来排成一个数,输出能排出的所有数字中最小的一个.例如输入数组{32, 321},则输出这两个能 ...
- 程序员面试题精选100题(51)-顺时针打印矩阵
// 程序员面试题精选100题(51)-顺时针打印矩阵.cpp : 定义控制台应用程序的入口点. //#include "stdafx.h" #include <iostre ...
- 程序员面试题精选100题:求从1到n的正数中1出现的次数
// 程序员面试题精选100题(25):求从1到n的正数中1出现的次数 // 如 f(253) = (2!=0) * 100 + 2 * f(99) + (5!=0) * 10 + 5 * f(9) ...
- 程序员面试题精选100题:11-40解题报告
程序员面试题精选100题(11)-求二元查找树的镜像[数据结构] 题目:输入一颗二元查找树,将该树转换为它的镜像,即在转换后的二元查找树中,左子树的结点都大于右子树的结点.用递归和循环两种方法完成 ...
- [程序员面试题精选100题]13.第一个只出现一次的字符
[题目] 在一个字符串中找到第一个只出现一次的字符.如输入abaccdeff,则输出b. [分析] [代码] /********************************* * 日期:2013- ...
最新文章
- Charles 从入门到精通
- 精通python能干嘛-一个程序员怎样才算精通Python?
- 微信支付-我遇到的那些问题
- 2016年计算机视觉和图像处理相关的国际会议一览表
- 如何做一场精彩的学术报告?
- 微软发布Azure Pipelines,开源项目可无限制使用CI/CD
- 如何获取上传文件的本地路径
- Apollo自动驾驶入门课程第⑩讲 — 控制(下)
- 软件工程师如何笑着活下去?
- python的简单GUI(多线程时钟)
- 叶氏量能大抄底(通达信副图)
- html pie标签,pie.html
- 基于SPringBoot的药品管理系统
- Flash 3D引擎比较
- laravel完全安装手册
- 脉搏测试报警系统设计
- 天池历届大赛答辩PPT及视频
- 5-2. 已有的小车实现samba文件共享
- 基于java的一款实时聊天系统,包含服务端 + 客户端 + web端
- 新员工加入企业微信的2种方法:邀请和被邀请
热门文章
- 区块链去中心化的生命之源:“DPOS(委托权益证明)共识机制”
- 如何在程序中生成崩溃转储dump文件以及如何分析dump
- 白话Elasticsearch04- 结构化搜索之使用terms query搜索多个值以及多值搜索结果优化
- Spring JDBC-自增键和行集RowSet
- python mvc web_浅谈python_web的MVC和MTV
- 2021-04-17 安装Ubuntu18.0.4 的深度学习训练服务器
- 华为鸿蒙产业链股票,华为P40或用鸿蒙 华为产业链概念股一览
- 前端python和go_Python_前端网页+前后端交互-Go语言中文社区
- vue中 .sync 修饰符 个人理解
- 基于matlab/simulink的双闭环PMSM控制