EOJ 1076 染气球 区间计数

题目描述

小强和旺财去逛街,他们去了欧尚,旺财看见超市里面的气球兴奋的直叫,呵呵,由于小强很喜欢旺财,所以下定决心给旺财买一打回家。回家小强数了下,一共有 N 只气球,小强将 N 只气球从左到右依次编号为 1、2、3……N,每次给 2 个整数 a,b (a<=b),小强便骑上他的“小飞鸽 ” 牌电动车从气球 a 开始到气球 b 依次给每个气球涂一次颜色。但是 N 次以后小强已经忘记了第 I 个气球已经涂过几次颜色了,你能帮他算出每个气球被涂过几次颜色吗?

输入格式

每个测试实例第一行为一个整数 N,(N <= 100000). 接下来的 N 行,每行包括 2 个整数 a b(1 <= a <= b <= N)。

当 N = 0,输入结束。

输出格式

每个测试实例输出一行,包括 N 个整数,第 I 个数代表第 I 个气球总共被涂色的次数。注意,每个数据之间有个空格,最后一个数据后面没空格。

样例

input

3
1 1
2 2
3 3
3
1 1
1 2
1 3
0

output

1 1 1
3 2 1

思路:暴力计数最坏情况可能会有n*n的复杂度,因此考虑只通过区间端点来计数。由于每次被区间覆盖一次,区间内的气球只会被涂色1次,因此每次区间出现其次对应的端点加1。为了避免左右端点统计重复,我们只统计左端点;为了表示区间的结束,我们对右区间之后的一个气球计数减1进行抵消,防止统计到没有覆盖到当前气球的区间。 最后只需遍历一遍数组,累计计数和即可,代码如下:

#include <iostream>
#include <cstring>

using namespace std;

int main()
{
    int n, a, b;
    long long sum;
    while(true){
        sum = 0;
        cin>>n;
        if(!n)
            break;
        int *c = new int[n+3];
        memset(c, 0, (n+3)*sizeof(int));
        for(int i=0; i<n; ++i){//端点计数
            cin>>a>>b;
            c[a]++;
            c[b+1]--;
        }
        for(int i=1; i<n; ++i){//统计计数
            sum += c[i];
            cout<<sum<<' ';
        }
        sum += c[n];
        cout<<sum<<endl;
    }
    return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值