Codeforces 85D Sum of Medians(线段树)

85D Sum of Medians

题目链接

题意:一个集合有添加,删除元素,每次查询输出集合位置为i % 5 == 3的位置和

思路:线段树,线段树记录下% 5 == 0, 1, 2, 3, 4的和,并且记录一个mov表示右移多少,每次添加一个值的时候,就当前位置之后的一整段位置都要右移一个单位,这样去搞线段树维护一下即可

代码:

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm>
using namespace std;

const int N = 100005;
typedef long long ll;

int n;
char str[10];
int op[N], opx[N];

struct Hash {
	int val, rank, id;
} hash[N];
int hn;

bool cmp(Hash a, Hash b) {
	return a.val < b.val;
}

#define lson(x) ((x<<1)+1)
#define rson(x) ((x<<1)+2)

struct Node {
	int l, r, mov;
 	ll k[5];
} node[N * 4];

void pushup(int x) {
	if (node[x].l == node[x].r) {
		ll tmp = 0;
		for (int i = 0; i < 5; i++) if (node[x].k[i]) tmp = node[x].k[i];
		if (tmp == 0) return;
		memset(node[x].k, 0, sizeof(node[x].k));
		node[x].k[node[x].mov % 5] = tmp;
		return;
 	}
	for (int i = 0; i < 5; i++)
		node[x].k[(i + node[x].mov) % 5] = node[lson(x)].k[i] + node[rson(x)].k[i];
}

void build(int l, int r, int x = 0) {
	node[x].l = l; node[x].r = r;
	node[x].mov = 0;
	memset(node[x].k, 0, sizeof(node[x].k));
	if (l == r) return;
	int mid = (l + r) / 2;
	build(l, mid, lson(x));
	build(mid + 1, r, rson(x));
}

void add(int l, int r, int v, int x = 0) {
	if (node[x].l >= l && node[x].r <= r) {
		node[x].mov += v;
		pushup(x);
		return;
 	}
 	int mid = (node[x].l + node[x].r) / 2;
 	if (l <= mid) add(l, r, v, lson(x));
 	if (r > mid) add(l, r, v, rson(x));
 	pushup(x);
}

int to[N];

void add2(int v, int val, int x = 0) {
	if (node[x].l == node[x].r) {
 		memset(node[x].k, 0, sizeof(node[x].k));
		if (val == 1)
  			node[x].k[node[x].mov % 5] = to[node[x].l];
		return;
 	}
	int mid = (node[x].l + node[x].r) / 2;
	if (v <= mid) add2(v, val, lson(x));
	if (v > mid) add2(v, val, rson(x));
	pushup(x);
}

int main() {
	hn = 0;
	scanf("%d", &n);
 	for (int i = 0; i < n; i++) {
	 	scanf("%s", str);
	 	if (str[0] == 'a' || str[0] == 'd') {
   			op[i] = 1;
   			if (str[0] == 'd') op[i] = -1;
   			scanf("%d", &opx[i]);
   			hash[hn].id = i;
   			hash[hn++].val = opx[i];
		} else op[i] = 0;
 	}
 	sort(hash, hash + hn, cmp);
 	to[0] = hash[0].val;
 	hash[0].rank = 0;
 	for (int i = 1; i < hn; i++) {
 		hash[i].rank = hash[i - 1].rank;
 		if (hash[i].val != hash[i - 1].val) {
    		hash[i].rank++;
    		to[hash[i].rank] = hash[i].val;
		}
  	}
  	build(0, hash[hn - 1].rank);
  	for (int i = 0; i < hn; i++)
  		opx[hash[i].id] = hash[i].rank;
	for (int i = 0; i < n; i++) {
		if (op[i] == 0) printf("%lld\n", node[0].k[3]);
		else {
  			add(opx[i], hash[hn - 1].rank, op[i]);
     		add2(opx[i], op[i]);
		}
 	}
	return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值
>