题目描述
小强和旺财去逛街,他们去了欧尚,旺财看见超市里面的气球兴奋的直叫,呵呵,由于小强很喜欢旺财,所以下定决心给旺财买一打回家。回家小强数了下,一共有 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;
}