【jzoj1746】【usaco 2013 feb Bronze】粉刷栅栏(Painting the Fence)

题目描述

题目:

农夫约翰最近正在将他的栅栏粉刷一下(这里所有的栅栏都是在一条直线上的)。他是这样来粉刷的:他从位置0出发,然后执行N条指令,例如,指令可以是“10 L”,表示约翰从当前的位置向左移动10个单位的距离,并且粉刷移动过程中遇到的栅栏,又或者是“15 R”,表示约翰从当前的位置向右移动15个单位的距离,并且粉刷移动过程中遇到的栅栏。

问题描述:

给定所有约翰需要移动的指令,请计算所有栅栏中至少被粉刷两次的栅栏的总长度。约翰最多远离初始位置1000000000个单位的距离。

输入
第一行一个正整数N。
接下来第2行到第N+1行,每行表示每条指令。

输出
只有一行一个整数,表示所有栅栏中至少被粉刷两次的栅栏的总长度。

样例输入

6
2 R
6 L
1 R
8 L
1 R
2 R

样例输出

6

数据范围限制
数据范围
:1<=N<=100000。

提示
说明
:样例中,有6个单位的长度至少被粉刷两次。分别是[-11,-8],[-4,-3],[0,2]。


解题思路

把每次粉刷都转化成区间,在根据不同情况做处理


#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
struct DT{
	int l,r;
}a[100100];
int n,x,L,R;
long long Gun;
char c;
bool cmp(const DT&k,const DT&l){
	return (k.l<l.l);
}
int main(){
	freopen("paint.in","r",stdin);
	freopen("paint.out","w",stdout);
	scanf("%d",&n);
	for(int i=1;i<=n;i++){
		scanf("%d",&x);
		cin>>c;
		if(c=='R'){//转换成区间
			a[i].l=L;
			a[i].r=L+x;
			L+=x;
		}else{
			a[i].l=L-x;
			a[i].r=L;
			L-=x;
		}
	} 
	sort(a+1,a+1+n,cmp);
	L=a[1].l,R=a[1].r;
	for(int i=2;i<=n;i++){
		if(a[i].r<L)continue;
		if(a[i].l>=L&&a[i].r<=R)//根据不同情况,计算重复长度
		   Gun+=a[i].r-a[i].l,L=a[i].r;
		 else 
		if(a[i].l>=L&&a[i].r>=R)
		   Gun+=R-a[i].l,L=R,R=a[i].r;
		 else
		if(a[i].l<=L&&a[i].r<=R)
		   Gun+=a[i].r-L,L=a[i].r;
		 else 
		if(a[i].l<=L&&a[i].r>=R)
		   Gun+=R-L,L=R,R=a[i].r;   
		 else 
		if(a[i].l>R&&a[i].r>R)
		   L=a[i].l,R=a[i].r;
	}      
	printf("%lld",Gun);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值