bzoj-2957 楼房重建

141 篇文章 0 订阅
88 篇文章 0 订阅

题意:

数轴上有n个楼,分别在1~n这些点上;

m次查询,每次改变一个楼的高度,问从(0,0)这个点可以看到多少楼;


题解:
对于一个楼来说要想看到这个楼,那么前面的楼的斜率一定比这个楼小;

那么考虑分块的话,就将块中楼的斜率都求出来;

然后维护出一个从块首元素开始的递增序列;

即包括块首元素的下标最小的序列;

扫一遍所有块,取该块之前的所有楼的最大斜率为ma;

在当前块中二分找比ma大的元素个数,并更新ma;

复杂度O(m*√n*log(√n)),时间基本和1s擦边;

但是BZ算的总时限所有可以无压力AC;


代码:


#include<math.h>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 110001
using namespace std;
int bk, h[N], cnt[400];
double k[N], q[400][400];
int main()
{
	int n, m, i, j, index, x, y, ans;
	double ma;
	scanf("%d%d", &n, &m);
	bk = sqrt(n);
	for (i = 1; i <= m; i++)
	{
		scanf("%d%d", &x, &y);
		h[x] = y;
		k[x] = (double)y / x;
		index = x / bk;
		for (j = index*bk, ma = 0, cnt[index] = 0; j <= index*bk + bk - 1; j++)
		if (k[j] > ma)
			q[index][++cnt[index]] = k[j], ma = k[j];
		ma = 0, ans = 0;
		for (j = 0; j <= n / bk; j++)
		{
			ans += cnt[j] - (upper_bound(q[j] + 1, q[j] + 1 + cnt[j], ma) - q[j] - 1);
			ma = max(ma, q[j][cnt[j]]);
		}
		printf("%d\n", ans);
	}
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值