和为S的两个数字O(logn * logn)解法

和为S的两个数字

http://www.nowcoder.com/questionTerminal/390da4f7a00f44bea7c2f3d19491311b

和为S的两个数字O(logn * logn)解法

题目描述

输入一个递增排序的数组和一个数字S,在数组中查找两个数,使得他们的和正好是S,如果有多对数字的和等于S,输出两个数的乘积最小的。

输出描述:

对应每个测试案例,输出两个数,小的先输出。

解法

1. 穷举:

对所有数据对都查找一遍
时间复杂度:O(n^2)

2. 哈希表:

建立哈希表查找 k-array[i]
时间复杂度:O(n)
空间复杂度:O(n)

3. 双指针:

时间复杂度:O(n),设置左右指针,向中间搜索合适的值,直到指针相遇
空间复杂度:O(1)

4. 双指针 + 二分查找:

时间复杂度:O(logn * logn)
空间复杂度:O(1)
前三种方法实现上难度为0,方法4需要对二分查找比较熟练。设置左右指针,每次向中间搜索时用二分法查找合适的值,没有合适的值就定位在闭边界位置。后台的测试用例应该都很短,这种情况下时间效率不如第一种方法。如果数组长度是1000,时间效率可提升10倍(10 = 2^10/(10 * 10))。
图片说明

代码

class Solution {
public:
    vector<int> FindNumbersWithSum(vector<int> array,int sum) {
            vector<int> result;
    if (!array.size())
        return result;
    int le = 0, ri = array.size() - 1;
    int l, r, m;
    int t;
    while (le < ri - 1) {
        if (array[le] + array[ri] == sum) {
            result = {array[le], array[ri]};
            return result;
        }
        else if (array[le] + array[ri] > sum) {
            t = sum - array[le];
            l = le + 1;
            r = ri;
            while (l < r) {
                m = (l + r) / 2;
                if (array[m] == t) {
                    result = {array[le], array[m]};
                    return result;
                }
                else if (array[m] > t)
                    r = m;
                else
                    l = m;
                if (l >= r - 1)
                    break;
            }
            ri = l;
        }
        else {
            t = sum - array[ri];
            l = le;
            r = ri - 1;
            while (l < r) {
                m = (l + r + 1) / 2;
                if (array[m] == t) {
                    result = {array[m], array[ri]};
                    return result;
                }
                else if (array[m] > t)
                    r = m;
                else
                    l = m;
                if (l >= r - 1)
                    break;
            }
            le = r;
        }
    }
    return result;

    }
};
全部评论

相关推荐

03-03 23:42
复旦大学 Java
_无论云泥意贯一:把复旦大学放前面,山东大学放后面,并且在两个大学后面标注985(用一些显眼的颜色标注)
点赞 评论 收藏
分享
04-16 10:27
已编辑
美团_Saas_后端开发
今天周一休息,突发奇想写一篇阶段总结。如题,我已经去了一个和Java彻底毫无关联的行业。曾经我以为自己能在计算机行业发光发热,拿到美团offer那会感觉自己天都亮了。没想到刚入行一年多就当了逃兵。从最开始的热爱到现在一看到代码就厌恶,不知道自己经历了什么。所以我去干什么了?答案是:在成都当了租房销售。上班那会压力大了就念叨着去干租房中介,但是一直下不去这个决心,想着自己学了四年多的计算机知识,终究还是不甘心。终于在某一天准备八股文的时候,看着无数篇和工作内容关系不大的理论知识,那一刻下定决心,决定尝试一下销售行业,也算是给自己一个交代。后面阴差阳错的投了成都自如去当租房管家,没想到面试很顺利,在当天一百多个面试的人里面,我成为了为数不多通过的几个幸运儿之一。目前已经培训通过,正式入职,也开了单,有压力但是每天过得很开心,真心喜欢那种和人交流的感觉,哪怕是最后没有选择找我租房。说这些也是想告诉那些大三,大四正在找Java实习而焦虑的同学:你们现在还年轻,选择很多,容错率也很高,可以尽情去尝试自己喜欢的行业和工作。不用因为某一次的面试没通过或者简历石沉大海而焦虑,更不用因为身边人都在挤编程的独木桥就强迫自己跟风。也算是自己的碎碎念吧,也希望自己能在新的领域取得一点小成就。也祝牛油工作顺利!
沉淀小子:干啥都不丢人啊,生存是必须要的,销售很考验一个人综合素质能力的,好的销售人脉和资源可不比写字楼的白领差啊
点赞 评论 收藏
分享
评论
2
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务