线段树 0 分求助
查看原帖
线段树 0 分求助
741244
Eason_cyx大愚若智楼主2024/12/5 13:57

rt。

#include <bits/stdc++.h>
#define int long long
using namespace std;
const long long INF = 1e18;
struct seg {
	unsigned sum; int l;
} tree[800005];
long long a[200005], n, m;
void pushup(int x) {
	tree[x].sum = tree[x*2].sum + tree[x*2+1].sum;
	if(tree[x*2].l == -1 || tree[x*2+1].l == -1) tree[x].l = -1;
	else {
		long long G = __gcd(tree[x*2].l, tree[x*2+1].l);
		if((tree[x*2].l / G) > (INF / tree[x*2+1].l)) {
			tree[x].l = -1;
		} else tree[x].l = tree[x*2].l / G * tree[x*2+1].l;
	}
} void build(int x, int l, int r) {
	if(l == r) {
		tree[x].sum = tree[x].l = a[l]; return ;
	} int mid = (l + r) >> 1;
	build(x*2, l, mid); build(x*2+1, mid+1, r);
	pushup(x);
} void change_gcd(int x, int l, int r, int y) {
	if(l == r) {
		tree[x].sum = __gcd(1ll * tree[x].sum, y); tree[x].l = tree[x].sum; return ;
	} int mid = (l + r) >> 1;
	if(tree[x*2].l != -1 && y % tree[x*2].l) change_gcd(x*2, l, mid, y);
	if(tree[x*2+1].l != -1 && y % tree[x*2+1].l) change_gcd(x*2+1, mid+1, r, y);
	pushup(x);
} void update(int x, int l, int r, int L, int R, int y) {
	if(L <= l && r <= R) {
		if(tree[x].l != -1 && y % tree[x].l) change_gcd(x, l, r, y);
		return ;
	} int mid = (l + r) >> 1;
	if(mid >= L) update(x*2, l, mid, L, R, y);
	if(mid < R) update(x*2+1, mid+1, r, L, R, y);
	pushup(x);
} unsigned query(int x, int l, int r, int L, int R) {
	if(l > R || r < L) return 0;
	if(L <= l && r <= R) {
		return tree[x].sum;
	} int mid = (l + r) >> 1;
	return query(x*2, l, mid, L, R) + query(x*2+1, mid+1, r, L, R);
} signed main() {
	cin >> n >> m;
	for(int i = 1;i <= n;i++) cin >> a[i]; build(1, 1, n);
	while(m--) {
		long long opt, l, r, x; cin >> opt >> l >> r;
		if(opt == 1) cin >> x, update(1, 1, n, l, r, x);
		else cout << query(1, 1, n, l, r) << endl;
	}
	return 0;
}
2024/12/5 13:57
加载中...