zoukankan      html  css  js  c++  java
  • 算法-最长子序列和C/C++实现(三个复杂度)

    最长子序列和的问题非常easy:

    就是一个数组,求出当中当中连续的某一段和,而这一段和是全部的连续段和的最大的值。求出这个值。


    先说复杂度最高的:O(n3)

    直接上代码,非常easy的:

    //
    //  main.cpp
    //  SumSequence
    //
    //  Created by Alps on 14-7-23.
    //  Copyright (c) 2014年 chen. All rights reserved.
    //
    
    #include <iostream>
    using namespace std;
    
    int MaxSubsequenceSum(const int A[], int N){
        int ThisSum, MaxSum, i, j, k;
        MaxSum = 0;
        for(i = 0; i < N; i++){
            for(j = i; j < N; j++){
                ThisSum = 0;
                for (k = i; k < j; k++) {
                    ThisSum += A[k];
                }
                MaxSum = ThisSum > MaxSum ? ThisSum: MaxSum;
            }
        }
        return MaxSum;
    }
    
    int main(int argc, const char * argv[])
    {
    
        int A[] = {1, 2, -5, 2, 5, 1, 8, -4};
        int N = sizeof(A)/sizeof(int);
    //    printf("%d
    ",N);
        int MaxSum = MaxSubsequenceSum(A, N);
        printf("%d
    ",MaxSum);
        return 0;
    }
    

    这个事实上非常easy,第一层for循环是i从头開始遍历。第二层for是j从i遍历到尾。第三层就是算i到j的这一段的和。

    时间复杂度是O(n3).

    以下说一个O(n2)的:

    代码例如以下:

    //
    //  main.cpp
    //  SumSequencen2
    //
    //  Created by Alps on 14-7-23.
    //  Copyright (c) 2014年 chen. All rights reserved.
    //
    
    #include <iostream>
    using namespace std;
    
    int MaxSubSequenceSum(const int A[], int N){
        int MaxSum, ThisSum, i, j;
        MaxSum = 0;
        for (i = 0; i < N; i++) {
            ThisSum = 0;
            for (j = i; j < N; j++) {
                ThisSum += A[j];
                MaxSum = MaxSum > ThisSum ? MaxSum : ThisSum;
            }
        }
        return MaxSum;
    }
    
    int main(int argc, const char * argv[])
    {
        int A[] = {1, 2, -5, 2, 5, 1, 8, -4};
        int N = sizeof(A)/sizeof(int);
        //    printf("%d
    ",N);
        int MaxSum = MaxSubSequenceSum(A, N);
        printf("%d
    ",MaxSum);
        
        return 0;
    }
    
    这个也比較好理解,第一层循环就是i从头到尾遍历,第二层循环是j从i遍历到尾,在遍历过程中不断检測ThisSum的大小,取Max(ThisSum, MaxSum)的数,并赋值给MaxSum,这样就能够知道MaxSum是多少了~


    另一个方法复杂度是O(nlogn)可是这个算法比較麻烦,代码也比較麻烦,我这里没有写~想学的能够去《数据结构与算法分析》来学习。


    这里有一个O(n)级别的算法来解决问题!!!:请看代码:


    //
    //  main.cpp
    //  SumSequencen
    //
    //  Created by Alps on 14-7-23.
    //  Copyright (c) 2014年 chen. All rights reserved.
    //
    
    #include <iostream>
    
    using namespace std;
    
    int MaxSubSequenceSum(const int A[], int N){
        int MaxSum, ThisSum, i;
        MaxSum = A[0];
        ThisSum = 0;
        for (i = 0; i < N; i++) {
            ThisSum += A[i];
            MaxSum = ThisSum > MaxSum ? ThisSum: MaxSum;
            if (ThisSum < 0) {
                ThisSum = 0;
                continue;
            }
        }
        return MaxSum;
    }
    
    int main(int argc, const char * argv[])
    {
        int A[] = {1, 2, -5, 2, 5, 1, 8, -4};
        int N = sizeof(A)/sizeof(int);
        //    printf("%d
    ",N);
        int MaxSum = MaxSubSequenceSum(A, N);
        printf("%d
    ",MaxSum);
        return 0;
    }
    

    O(n)级别的这类算法算是比較完美的算法了。我对这个算法的理解就是,在一个数组里,有非常多非常多段,这些段都有一个和,最小的段是一个元素,而最大的序列和肯定是一个段,或者是两个段的和,和就是加上一个正数就变大,所以当一个段是负数的时候,我就直接抛弃掉了~(除非全部都是负数,就找一个最大的。)

    所以就有了上面的算法。。不懂的请留言~


  • 相关阅读:
    03-19总结
    JS-DOM
    难题汇总,浮动,伪元素(行级),
    bug汇总
    两栏布局
    表单 form
    a 、ul、 table 标签
    Bootstrap的基本使用(css、js文件的引入)
    6. CSS样式
    5 CSS
  • 原文地址:https://www.cnblogs.com/lcchuguo/p/4512921.html
Copyright © 2011-2022 走看看