您的位置:首页 > 其它

Ural_1303

2011-08-21 15:52 232 查看
区间覆盖问题,选取最少的线段,覆盖一个区间.

首先,区间外的线段不予考虑,按线段左点排序,从起点开始覆盖,在左点在起点左边的点中,选取右点最右的点,之后将该右点作为新的起点直至覆盖整个区间为止,如果找不到新的线段右点在起点右边,说明不能完全覆盖

#include <cstdio>
#include <string>
#include <math.h>
#include <string.h>
#include <stdlib.h>
#include <iostream>
#include <algorithm>
using namespace std;
int m,p,a,b;

struct xd{
int s,e;
bool operator < (const xd& a)const{//按照开始坐标排序
return s<a.s;
}
}ss[100005];
int vis[100005];//标记是否在结果中
int main(){
p=0;
memset(vis,0,sizeof vis);

scanf("%d",&m);
while(scanf("%d%d",&a,&b)){
if(a==0&&b==0)break;
if(b<=0||a>=m||a==b)continue;
ss[p].s=a,ss[p].e=b;
p++;
}

sort(ss,ss+p);

int flag=1,ed=0,pos=0,tp=0;

while(ed<m){
int ned=0,ni=0;
while(pos<p&&ss[pos].s<=ed){
if(ss[pos].e>ned){//在起始点在该点左边的点中,选择终止点最靠右的点
ned=ss[pos].e;
ni=pos;
}
pos++;
}
if(ned==0){flag=0;break;}//没有增量,失败,退出

ed=ned;//将该点作为新的起始点并标记结果
vis[ni]=1;
tp++;
}

if(p==0||flag==0){
printf("No solution\n");
}else{
printf("%d\n",tp);
for(int i=0;i<p;i++){
if(vis[i])printf("%d %d\n",ss[i].s,ss[i].e);
}
}
system("pause");
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: