博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
HUT-XXXX Money out of Thin Air 线段树
阅读量:6325 次
发布时间:2019-06-22

本文共 3059 字,大约阅读时间需要 10 分钟。

该题题义是给定一个公司的结构,然后针对每个员工的一系列操作。由于问题牵涉到一个员工以及一个员工所领导的部门,因此用普通线性结构显然效率太低。这里用到了线段树。

step1:首先我们要确定题目是要对给定的点或者是区间进行操作,那么我们自然而然会想到线段树,但是这题还不是直接的线段树,需要对题中给定的关系进行离散话,也就是按

    照先序或者是后序遍历进行排序,这样某一个员工所领导的部门下的成员就在物理上连续了,并且我们用两个数组分别记录某个成员所领导的部门的左边界和右边界。

step2:有了上一步的操作后,我们就可以进行构树了,按照普通的线段树进行构建,线段树仅仅只是我们处理点和区间的一种数据结构,其本身没有什么实际意义,记得用一个映

    射将线段树的点和实际员工的编号联系起来,这样方便了初始化,也便于最后的输出。

step3:此时我们就可以来处理数据中所给定的操作了,点更新和区间更新,当然这里可以用到lazy标记。

PS:由于整个数很可能退化成链表,所以最好使用非递归来实现先序或者是后序遍历。

代码如下:

#include 
#include
#include
#include
#include
#define MAXN 50005using namespace std;int N, Q, s[MAXN], stk[MAXN], visit[MAXN], top;int seq[MAXN], idx, L[MAXN], R[MAXN];typedef long long int Int64;queue
q[50005];struct Node{ int l, r; Int64 sum, lazy;}e[200005];void getseq(int boss) { int pos; idx = -1; top = 1; stk[top] = boss; memset(visit, 0, sizeof (visit)); while (top) { pos = stk[top]; if (!visit[pos]) { visit[pos] = 1; seq[++idx] = pos; L[pos] = idx; } if (!q[pos].empty()) { stk[++top] = q[pos].front(); q[pos].pop(); } else { R[pos] = idx; --top; } }}void build(int p, int l, int r){ e[p].l = l, e[p].r = r; e[p].lazy = 0; if (l!= r) { int mid = (l + r) >> 1; build(p<<1, l, mid); build(p<<1|1, mid+1, r); e[p].sum = e[p<<1].sum + e[p<<1|1].sum; } else { // l == r e[p].sum = s[seq[l]]; }}inline void update(int p){ e[p].sum = e[p<<1].sum + e[p<<1|1].sum;}Int64 sum(int p, int l, int r){ if (e[p].l == l && r == e[p].r) { return e[p].sum; } int mid = (e[p].l + e[p].r) >> 1; if (e[p].lazy) { e[p<<1].lazy += e[p].lazy; e[p<<1|1].lazy += e[p].lazy; e[p<<1].sum += (e[p<<1].r-e[p<<1].l+1)*e[p].lazy; e[p<<1|1].sum += (e[p<<1|1].r-e[p<<1|1].l+1)*e[p].lazy; e[p].lazy = 0; } if (r <= mid) { return sum(p<<1, l, r); } else if (l > mid) { return sum(p<<1|1, l, r); } else { return sum(p<<1, l, mid) + sum(p<<1|1, mid+1, r); }}void modify(int p, int l, int r, Int64 x){ if (e[p].l == l && r == e[p].r) { e[p].lazy += x; e[p].sum += (e[p].r-e[p].l+1) * x; // 身上有延时标记,但是本身的值一定要是正确的 return; } int mid = (e[p].l + e[p].r) >> 1; if (e[p].lazy) { e[p<<1].lazy += e[p].lazy; e[p<<1|1].lazy += e[p].lazy; e[p<<1].sum += (e[p<<1].r-e[p<<1].l+1)*e[p].lazy; e[p<<1|1].sum += (e[p<<1|1].r-e[p<<1|1].l+1)*e[p].lazy; e[p].lazy = 0; } if (r <= mid) { modify(p<<1, l, r, x); } else if (l > mid) { modify(p<<1|1, l, r, x); } else { modify(p<<1, l, mid, x); modify(p<<1|1, mid+1, r, x); } update(p);}int main(){ int p, x, y, z, first = 1; Int64 level; char op[15]; while (scanf("%d %d %d", &N, &Q, &s[0]), N|Q|s[0]) { if (first) { first = 0; } else { puts(""); } for (int i = 1; i < N; ++i) { scanf("%d %d", &p, &s[i]); q[p].push(i); } getseq(0); build(1, 0, N-1); // 区间划分是 0 - N-1 while (Q--) { scanf("%s %d %d %d", op, &x, &y, &z); if (op[0] == 'e') { level = sum(1, L[x], L[x]); if (level < y) { modify(1, L[x], L[x], z); } } else { level = sum(1, L[x], R[x]); if (1.*level/((R[x]-L[x]+1)) < y) { modify(1, L[x], R[x], z); } } } for (int i = 0; i < N; ++i) { printf("%lld\n", sum(1, L[i], L[i])); } } return 0;}

转载地址:http://ezmaa.baihongyu.com/

你可能感兴趣的文章
CSS变量variable
查看>>
MRF能量优化
查看>>
JForum 源码分析
查看>>
【nginx】nginx:利用负载均衡原理实现代码的热部署和灰度发布
查看>>
电子书 VS 纸质书
查看>>
老板雇佣的是40小时的工作量还是普通人需要花40小时才能完成的工作成果?
查看>>
javascript实时保存时出现改动多条记录的bug
查看>>
leetCode 57.Insert Interval (插入区间) 解题思路和方法
查看>>
VREP中的力触觉设备接口(CHAI3D)
查看>>
jQuery中append、insertBefore、after与insertAfter方法注意事项
查看>>
Android零基础入门第1节:Android的前世今生
查看>>
【Linux】 Linux权限管理与特殊权限
查看>>
Android的事件分发
查看>>
【Lua】环境安装与HelloWorld
查看>>
maven多module项目中千万不要引入其它模块的单元測试代码
查看>>
Java集合框架:EnumMap
查看>>
Oracle之外键(Foreign Key)使用方法具体解释(二)- 级联删除(DELETE CASCADE)
查看>>
python学习:收集ip信息
查看>>
MySQL之IDE工具介绍及数据备份
查看>>
poj1190生日蛋糕
查看>>