序列查询新解 CSP 202112-2

 

 

 

 思路:

这道题搞得我好头大,一开始以为是前缀和,在1e5范围内没有问题,能拿70分,后面的就处理不了了。后来看了y总讲解,才有新的思路。

现在有两个函数fx和gx,要计算他们差值的绝对值,根据我们上一题的思路,我们还是把N加到a[n+1]去,采用遍历每一个区间,算出差值,但在每一个区间里,fx和gx的关系有3种。

1.fx全大于gx

2.gx全大于fx

3.由于gx单调递增,左边小于fx,右边大于fx

根据这三种情况,我们可以分类讨论,由于要遍历每一个区间,我们要分情况把每一个区间里的fx和gx求出来,然后一减得到差值,取绝对值加起来。

AC代码:

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N = 100010;
int n, m;
int a[N];
int R;

LL get(int l, int r) {//得到g(x)在l到r区间的和
    //如果就只有一段
	if (l / R == r / R)return (LL)(r - l + 1) * (l / R);
    //由于g(x)是一个等差数列,用高斯公式算
	int a = l / R + 1, b = r / R - 1;
	LL res = (a + b) * (LL)(b - a + 1) / 2 * (LL)R;
    //左边小段计算
	res += (l / R) * (LL)(a * R - l);
    //右边小段计算
	res += (b + 1) * (LL)(r - (b * R + R)+1);
	return res;
}

int main() {
	cin >> n >> m;
	for (int i = 1; i <= n; i++)scanf("%d", &a[i]);
	a[n + 1] = m;
	R = m / (n + 1);
	LL res = 0;
	for (int i = 0; i <= n; i++) {
        //这一区间的左右端点
		int l = a[i], r = a[i + 1] - 1;
        //左右端点对应的g(x)值
		int x = l / R, y = r / R;
		if (y <= i || x >= i) {//全在上或全在下
			res += abs((LL)i * (r - l + 1) - get(l, r));
		}
		else {//分段,l到mid,mid+1到r
			int mid = i * R;//交点
			res += abs((LL)i * (mid - l + 1) - get(l, mid));
			res += abs((LL)i * (r - mid) - get(mid + 1 , r));
		}
	}
	cout << res;
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

星河边采花

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值