码迷,mamicode.com
首页 > 编程语言 > 详细

(算法)最长重叠线段或区间

时间:2015-05-29 00:34:03      阅读:472      评论:0      收藏:0      [点我收藏+]

标签:

题目:

X轴上有N条线段,每条线段包括1个起点和终点。线段的重叠是这样来算的,[10 20]和[12 25]的重叠部分为[12 20]。
给出N条线段的起点和终点,从中选出2条线段,这两条线段的重叠部分是最长的。输出这个最长的距离。如果没有重叠,输出0。
 

思路:

1、暴力计算

依次计算两两线段之间的重叠长度,但复杂度太高

2、动态规划

假设S[n]表示n条线段中最长重叠距离,最长重叠距离只与两条线段有关,那么考虑两种情况:

1. 最长重叠距离与第n条线段无关,则最长重叠距离存在于前n-1条线段中,即S[n]=S[n-1];

2. 最长重叠距离与第n条线段有关,则最长重叠距离为第n条线段与前面n-1条线段中的最大重叠距离者,S[n]=max(overlap(segment[n],segment[i])), i=1....n-1

因此得到状态转移方程:

S[n]=0; n<=1

S[2]=overlap(segment[0],segment[1])

S[n]=max(S[n-1],max(overlap(segment[n],segment[i])))

代码:

#include <iostream>
#include <algorithm>

using namespace std;


typedef struct{
    int start;
    int end;
}segment;

bool isShorter(const segment &s1,const segment &s2);
segment commonSegment(const segment &seg_a,const segment &seg_b);
int findLongestSegment(const vector<segment> &segments,int size,segment &maxSegment);

int main()
{
    int n;
    int start,end;
    while(cin>>n){
        vector<segment> segments(n);
        for(int i=0;i<n;i++){
            if(cin>>start && cin>>end){
                segments[i].start=start;
                segments[i].end=end;
            }
        }

        // sort by segment.end
        sort(segments.begin(),segments.end(),isShorter);

        segment maxSeg;
        int maxLen;

        maxLen=findLongestSegment(segments,n,maxSeg);

        cout<<"Longest Length:"<<maxLen<<endl;
        cout<<"start:"<<maxSeg.start<<endl;
        cout<<"end:"<<maxSeg.end<<endl;
    }

    return 0;
}

bool isShorter(const segment &s1,const segment &s2){
    return s1.end<s2.end;
}


//assume seg_a.end<seg_b.end
segment commonSegment(const segment &seg_a,const segment &seg_b){
    segment commonSeg;
    if(seg_a.end<seg_b.start){
        commonSeg.start=0;
        commonSeg.end=0;
    }
    else{
        commonSeg.start=seg_b.start;
        commonSeg.end=seg_a.end;
    }
    return commonSeg;
}

int findLongestSegment(const vector<segment> &segments,int size,segment &maxSegment){
    if(size<=0)
        return 0;
        /*
    if(size==1){
        maxSegment.start=segments[0].start;
        maxSegment.end=segments[0].end;
        return maxSegment.end-maxSegment.start;
    }
        */
    if(size==2){
        maxSegment=commonSegment(segments[0],segments[1]);
        return maxSegment.end-maxSegment.start;
    }

    // size>2
    // dynamic programming
    int maxLen,tmpLen;
    segment tmpMaxSeg;
    maxLen=findLongestSegment(segments,size-1,tmpMaxSeg);
    maxSegment=tmpMaxSeg;

    // maxSegment=(maxSegment,commonSeg)
    segment commonSeg;
    for(int i=0;i<size-1;i++){
        commonSeg=commonSegment(segments[i],segments[size-1]);
        tmpLen=commonSeg.end-commonSeg.start;
        if(tmpLen>maxLen){
            maxLen=tmpLen;
            maxSegment=commonSeg;
        }
    }

    return maxSegment.end-maxSegment.start;
}

运行结果:

技术分享

 

(算法)最长重叠线段或区间

标签:

原文地址:http://www.cnblogs.com/AndyJee/p/4537376.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!