问题描述:https://leetcode.com/problems/two-sum/

思路1:暴力搜索

根据排列组合原理,列举Cn取2对数字,逐对进行判断,效率是O(n^2-1/2n),代码如下:

var twoSum = function(nums, target) {
for (let i = 0; i != nums.length; ++i) {
for (let j = i + 1; j != nums.length; ++j) {
if (nums[i] + nums[j] == target) {
return [i, j];
}
}
}
};

思路2:聪明一点的搜索

先将数组排序,这个nlgn内可以完成,然后弄两个下标分别指向头和尾,边做判断边根据判断结果缩小搜索范围,这样一般在扫描完整个数组之前(n以内)就可以找到相应元素了,不过需要额外的空间来保存元素和原下标间的映射关系。最终效率是O(nlgn):

// O(nlgn)
var twoSum2 = function(nums, target) {
let copy = nums.map((x, i) => {
return {
val: x,
index: i,
};
}); // 保存元素的下标值
copy.sort((a, b) => a.val - b.val); // 按照元素值从小到大排序 let i = 0, j = copy.length - 1;
let sum;
while ((sum = copy[i].val + copy[j].val) != target) {
sum > target ? j-- : i++;
} i = copy[i].index;
j = copy[j].index;
return i < j ? [i, j] : [j, i];
};
正确性我也是纠结了一会儿,毕竟数学很菜。。。所以给出一个乱七八糟。。又不严谨的。。。“证明”↓

我们证明这个循环不变式:按照从有序数组的外围朝向内侧的搜索顺序,如果(sum = copy[i].val + copy[j].val) != target那么要找的元素一定在i+1..j或者i..j-1范围中,至于变化i还是j取决于copy[i].val + copy[j].val相对target是大了还是小了:大了一定减j,小了一定加i。

初始化:单看i=0以及j=copy.length - 1;的话是显然成立的,不过这属于特殊情况,属于没路走了,只能i+1或者j-1。选取一个稍普通些的情形i+1..j-1(i=0且j=copy.length - 1),这个时候i和j都是可进可退的,那么当sum > target的时候为什么一定是j-1-1而不是i+1-1呢(这两种操作都会让sum更接近target)?一个直觉且没毛病的理由是i..j-1在上个(或者上上个)迭代已经判断过了,所以此路明显不通,所以就只能是j-1-1了。然而我们需要考察更普通的情形,连续n次迭代sum都小于target,于是i先加了n,之后m次迭代sum都大于target,j才随后减去m,那么在j逐步-1的过程中i可不可以尝试回退一步(-1,回退n步和1步情形都是一样的,假设回退一步方便讨论)呢?因为这么做确实也让sum变小了,并且容易知道i+n-1..j-m这对数并没有判断过(i累计加了n的时候j仍可能还在原地踏步)。不妨假设[i+n-1, j-m]就是原问题的解,即i前进到i+n又回退到i+n-1并且copy[i+n-1].val + copy[j-m].val == target成立。仔细考察“有序数组”这一前置条件,会发现倘若i和j是原问题的解,也就是copy[i].val + copy[j].val == target成立,那么copy[i].val + copy[任何大于j的下标值].val总是大于target,这是因为copy[任何大于j的下标值].val > copy[j].val,言下之意,考虑最终的情形,必定i和j是有一方会先到达要找的元素之一且在原地等待,不存在什么走过头再回头的情形,于是就推翻假设了。。。此时已经考虑了所有情况,因此循环不变式成立。当sum < target时类似。

保持:假设对于某个i..j(i>0且j<copy.length - 1)循环不变式成立,也就是要找的元素一定在i+1..j或者i..j-1中。那么再进行一次迭代,也就是在i+1..j或者i..j-1中搜索,这又回到了初始化的情形,所以循环不变式仍然成立。

终止:因为题目确保了一定有那么一对元素存在,迭代终止的时候就是sun == target找到答案的时候。

思路3:利用HashMap巧解

事实上在给出target和一个元素后,另外一个元素就已经随之确定了。因此可以构建一个map,用另外一个元素的值作为key(或者自己本身的值),而value部分可以用来存当前元素的下标,这样一来,只需要至多扫描一遍数组(或者一遍以内),就可以得到答案了。效率是O(n)。js代码如下:

// O(n)
var twoSum3 = function(nums, target) {
let obj = {};
nums.forEach((x, i) => obj[target - x] = i);
for (let i = 0; i != nums.length; ++i) {
let j = obj[nums[i]];
if (j != undefined && j != i) {
return i < j ? [i, j] : [j, i];
}
}
}; // O(n),渐进性没有改善,不过去掉了一些尾巴
// Runtime: 52 ms, faster than 100.00% of JavaScript online submissions for Two Sum.
var twoSum4 = function(nums, target) {
let obj = {}, j; // 用obj作为map
for (let i = 0; i != nums.length; ++i) {
if ((j = obj[nums[i]]) != undefined) {
return [j, i];
}
obj[target - nums[i]] = i;
}
}; // 这个和上一个是一样的,纯粹为了测试一下用obj作map快还是new Map()快。结果是慢了一点。
var twoSum5 = function(nums, target) {
let map = new Map();
let obj = {}, j;
for (let i = 0; i != nums.length; ++i) {
if ((j = map.get(nums[i])) != undefined) {
return [j, i];
}
map.set(target - nums[i], i);
}
};

LeetCode #001# Two Sum(js描述)的更多相关文章

  1. 【JAVA、C++】LeetCode 001 Two Sum

    Given an array of integers, find two numbers such that they add up to a specific target number. The ...

  2. [Leetcode][001] Two Sum (Java)

    题目在这里: https://leetcode.com/problems/two-sum/ [标签]Array; Hash Table [个人分析] 这个题目,我感觉也可以算是空间换时间的例子.如果是 ...

  3. LeetCode 算法题解 js 版 (001 Two Sum)

    LeetCode 算法题解 js 版 (001 Two Sum) 两数之和 https://leetcode.com/problems/two-sum/submissions/ https://lee ...

  4. 【LeetCode】1022. Sum of Root To Leaf Binary Numbers 解题报告(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 日期 题目地址:https://leetco ...

  5. 【LeetCode】985. Sum of Even Numbers After Queries 解题报告(C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 暴力 找规律 日期 题目地址:https://lee ...

  6. 【LeetCode】129. Sum Root to Leaf Numbers 解题报告(Python)

    [LeetCode]129. Sum Root to Leaf Numbers 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://leetcode.com/pr ...

  7. Java for LeetCode 216 Combination Sum III

    Find all possible combinations of k numbers that add up to a number n, given that only numbers from ...

  8. LeetCode 1 Two Sum 解题报告

    LeetCode 1 Two Sum 解题报告 偶然间听见leetcode这个平台,这里面题量也不是很多200多题,打算平时有空在研究生期间就刷完,跟跟多的练习算法的人进行交流思想,一定的ACM算法积 ...

  9. [leetCode][013] Two Sum 2

    题目: Given an array of integers that is already sorted in ascending order, find two numbers such that ...

随机推荐

  1. GitLab push除发Jenkins事件

    1.打开Jenkins项目配置 2.勾选Trigger builds remotely (e.g., from scripts) 3.Authentication Token随便填个内容(比方1234 ...

  2. Ubuntu下开启Mysql远程访问的方法

    首先想说,JetProfiler对分析项目中MySQL问题以及优化,是个非常好的工具.但是看网上文章,中文介绍真的不多.是因为国内现在都不用MySQL了吗? 因为公司JetProfiler是共用的,安 ...

  3. Springboot中enable注解

    这句话可以作为理解springboot自动注入的原理的钥匙:ImportSelector接口的selectImports返回的数组(类的全类名)都会被纳入到spring容器中. 至于spring怎么根 ...

  4. thinkphp5中使用PHPExcel(转载)

    thinkphp5中可以使用composer来获取第三方类库,使用起来特别方便,例如:可是使用composer下载PHPMailer,think-captcha(验证码)等等…… 接下来说一下怎么使用 ...

  5. java学习(一)--- 基础语法

    学习内容来 自菜鸟教程 http://www.runoob.com/java/java-object-classes.html   Java基础 Java:一个Java程序可以认为是一系列的对象组合, ...

  6. Python3.6 性能测试框架Locust的搭建与使用

    背景 Python3.6 性能测试框架Locust的搭建与使用 基础 python版本:python3.6 方法一: pip install locustio 方法二: 开发工具:pycharm 使用 ...

  7. python中字符串格式化的四种方法

    name = "huangemiling" age= 10 address = 'nanjing' print("My name is %s,age is %d,I co ...

  8. Intellij IDEA快捷键与使用技巧一览表

    Intellij IDEA快捷键 Ctrl+Shift + Enter,语句完成 "!",否定完成,输入表达式时按 "!"键 Ctrl+E,最近的文件 Ctrl ...

  9. Nginx负载均衡后端健康检查

    参考文档:https://www.cnblogs.com/kevingrace/p/6685698.html 本次使用第三方模块nginx_upstream_check_module的,要使用这个第三 ...

  10. CodeForces 733B Parade

    B. Parade time limit per test1 second memory limit per test256 megabytes inputstandard input outputs ...