57 和为S的数字

Posted guoxinxin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了57 和为S的数字相关的知识,希望对你有一定的参考价值。

题目一:和为S的两个数字

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

输出描述:

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

测试序列

1)功能测试(数组中存在和为s的两个数;数组中不存在和为s的两个数)

2)特殊输入测试(数组指针为空指针)

解题思路:

使用两个指针分别指向首尾,然后根据和不断向中间逼近,遇到的第一个满足条件的两个数,就是乘积最小的(原理同 正方形、矩阵周长相同,正方向面积大 可知,4*4>2*6)。

class Solution {
public:
    vector<int> FindNumbersWithSum(vector<int> array,int sum) {
        int size = array.size();
        vector<int> res;
        if(size<2)
            return res;
        
        int front = 0, back = size-1;
        
        while(back>front){
            int s = array[back]+array[front];
            if(s==sum){
                res.push_back(array[front]);
                res.push_back(array[back]);
                break;
            }else if(s<sum){
                front++;
            }else{
                back--;
            }
        }
        
        return res; //没有找到,直接返回空的res
        
    }
};

  

题目二:和为S的连续正数序列

小明很喜欢数学,有一天他在做数学作业时,要求计算出9~16的和,他马上就写出了正确答案是100。但是他并不满足于此,他在想究竟有多少种连续的正数序列的和为100(至少包括两个数)。没多久,他就得到另一组连续正数和为100的序列:18,19,20,21,22。现在把问题交给你,你能不能也很快的找出所有和为S的连续正数序列? Good Luck!

输出描述:

输出所有和为S的连续正数序列。序列内按照从小至大的顺序,序列间按照开始数字从小到大的顺序

测试序列

1)功能测试(存在和为s的连续序列;不存在和为s的连续序列)

2)边界值测试(连续序列的最小和3)

解题思路:

仍然使用双指针,记录序列的首尾(最大值与最小值)

如果从front到back序列和大于sum,则可以从序列中去掉较小的值,即增大front的值

如果从front到back序列和小于sum,则可以增大back的值,让序列包含更多的数

因为序列中至少有两个数字,因此一直增加front到(1+sum)/2为止

class Solution {
public:
    vector<vector<int> > FindContinuousSequence(int sum) {
        vector<vector<int> >  res;
        if(sum<3)    //两个正数的和至少是3
            return res;
        
        int front = 1, back=2; //不能初始化为 0/1
        int mid = (1+sum)/2;
        //while(front<back && back<sum){
        while(front<mid){
            int s = getSeqSum(front, back); 
            if(s==sum){
                vector<int> line;
                for(int i=front;i<=back;i++){
                    line.push_back(i);
                }
                res.push_back(line);
                front++;  //重新找
            }else if(s<sum){
                back++;
            }else{
                front++; //并更新back
                //不用更新back!!
                //back--; //会造成front==back的情况,然后不满足while循环
                //back=front+1;  //有比较好的更新么 
            }
        }
        return res;
    }
    
    int getSeqSum (int front, int back){
        int ssum = (front+back)*(back-front+1)/2;
        return ssum;
    }
};

  

 

以上是关于57 和为S的数字的主要内容,如果未能解决你的问题,请参考以下文章

57 和为S的数字

剑指OFFER----面试题57. 和为s的两个数字

LeetCode(剑指 Offer)- 57. 和为 s 的两个数字

LeetCode(剑指 Offer)- 57. 和为 s 的两个数字

面试题57. 和为s的两个数字

面试题57. 和为s的两个数字