简单数据结构题(from 钟子谦——IOI2018集训队自选题)

试题描述

给一棵 \(n\) 个点的树,点权开始为 \(0\) ,有 \(q\) 次操作,每次操作是选择一个点,把周围一圈点点权 \(+1\)(一个点周围的点为与该点距离为 \(1\) 的点),在该操作后你需要输出当前周围一圈点点权的异或和。

由于输出量较大,设第 \(i\) 个询问输出为 \(ans_i\),你只需要输出

\begin{equation}

[\sum^q_{i=1}ans_i \cdot (i^2+i)] \texttt{ mod } (10^9+7)

\notag

\end{equation}

输入

第一行两个数 \(n\) , \(q\) ,表示树的点数和操作数。

接下来 \(n-1\) 行每行两个数表示树上的一条边。

接下来 \(q\) 行每行一个数 \(x\),表示把 \(x\) 周围一圈点点权 \(+1\)。

输出

输出一个 \([0,10^9+7)\) 的数,详见题目描述。

输入示例

5 10
1 2
2 3
2 4
3 5
1
5
2
4
3
5
4
2
3
1

输出示例

2060

数据规模及约定

对于 \(80\texttt{%}\) 的数据,保证 \(n = 1000\)

对于 \(90\texttt{%}\) 的数据,保证 \(n = 100000\)

对于 \(100\texttt{%}\) 的数据,保证 \(n = 500000\)

题解

这题差最后一步想到了。

不过好像除了 trie 树这步最妙这题也没其他什么了。。。

每个点周围的点可以分为父节点和子树中的一层节点,所以我们可以把一个节点 \(u\) 的所有儿子的信息存储到节点 \(u\) 中,然后对于一个询问,父亲单独计算,统一计算所有儿子的。

所以现在要一个可以支持删除、插入(对父亲的单独操作)、全体加 \(1\)、求全体异或和的数据结构。

一个数 \(x\) 加上 \(1\) 可以看做 \(x \rightarrow x \bigoplus (2 \cdot lowbit(\texttt{~}x) - 1)\)(\(\bigoplus\) 表示异或运算符,\(\texttt{~}\) 表示取反运算符,\(lowbit(t)\) 表示只取 \(t\) 最低位的 \(1\)),所以我们需要维护集合内所有 \(x\) 的同时维护所有的 \(lowbit(\texttt{~}x)\),确切地,只需要知道每种 \(lowbit(x)\)(最多 \(\log q\) 种)的取值有多少个。

然后就是我没想到的最后一步了:建立 trie 树保存集合内所有数(从浅到深按从低位到高位的顺序保存),\(lowbit(\texttt{~}x)\) 很好得到了,然后全体 \(+1\) 就是每次交换左右子树,然后递归到左子树去(即最低位异或 \(1\) 然后处理进位)。

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cctype>
#include <algorithm>
using namespace std;
#define rep(i, s, t) for(int i = (s); i <= (t); i++)
#define dwn(i, s, t) for(int i = (s); i >= (t); i--) int read() {
int x = 0, f = 1; char c = getchar();
while(!isdigit(c)){ if(c == '-') f = -1; c = getchar(); }
while(isdigit(c)){ x = x * 10 + c - '0'; c = getchar(); }
return x * f;
} #define maxn 500010
#define maxm 1000010
#define maxnode 10000010
#define maxlog 25
#define MOD 1000000007
#define LL long long int n, m, head[maxn], nxt[maxm], to[maxm], Fa[maxn], at[maxn];
int ToT, rt[maxn], val[maxnode], ch[maxnode][2], siz[maxnode], fa[maxnode], rec[maxlog], cntr; void AddEdge(int a, int b) {
to[++m] = b; nxt[m] = head[a]; head[a] = m;
swap(a, b);
to[++m] = b; nxt[m] = head[a]; head[a] = m;
return ;
} int getnode() {
int u;
if(cntr) u = rec[cntr--];
else u = ++ToT;
ch[u][0] = ch[u][1] = fa[u] = siz[u] = 0;
return u;
}
int Insert(int& r, int v) {
if(!r) r = getnode(), val[r] = v;
else val[r] ^= v;
int u = r; siz[u]++;
rep(i, 1, 20) {
int x = v & 1; v >>= 1;
if(!ch[u][x]) fa[ch[u][x] = getnode()] = u;
u = ch[u][x]; siz[u]++;
}
// printf("Insert(%d): %d\n", r, u);
return u;
} void build(int u) {
rt[u] = ++ToT;
for(int e = head[u]; e; e = nxt[e]) if(to[e] != Fa[u]) {
at[to[e]] = Insert(rt[u], 0);
Fa[to[e]] = u;
build(to[e]);
}
return ;
} int getnum(int u) {
int res = 0; cntr = 0;
while(u) {
if(fa[u] && ch[fa[u]][1] == u) res = res << 1 | 1;
else if(fa[u]) res <<= 1;
// printf("getnum: %d\n", u);
siz[u]--;
if(!siz[u] && fa[u]) {
rec[++cntr] = u;
// printf("recycle %d\n", u);
if(ch[fa[u]][1] == u) ch[fa[u]][1] = 0;
else ch[fa[u]][0] = 0;
}
if(!fa[u]) val[u] ^= res;
u = fa[u];
}
return res;
}
void Add(int r) {
int u = r, d = 0;
while(u) {
int s = ch[u][0] ? siz[ch[u][0]] : 0;
if(!ch[u][1]) s = siz[u];
if(s & 1) val[r] ^= (1 << d + 1) - 1;
u = ch[u][1]; d++;
}
u = r;
while(u) {
swap(ch[u][0], ch[u][1]);
u = ch[u][0];
}
return ;
} int main() {
n = read(); int q = read();
rep(i, 1, n - 1) {
int a = read(), b = read();
AddEdge(a, b);
} Insert(rt[0], 0);
build(1);
int Ans = 0;
rep(i, 1, q) {
int u = read(), ans = 0;
if(Fa[u]) {
ans = getnum(at[Fa[u]]) + 1;
// printf("faans: %d\n", ans);
at[Fa[u]] = Insert(rt[Fa[Fa[u]]], ans);
}
// printf("beforeAdd: %d\n", val[rt[u]]);
Add(rt[u]);
ans ^= val[rt[u]];
// printf("afterAdd: %d\n", val[rt[u]]);
(Ans += (LL)ans * ((LL)i * i % MOD + i) % MOD) %= MOD;
}
printf("%d\n", Ans); return 0;
}

简单数据结构题(from 钟子谦——IOI2018集训队自选题)的更多相关文章

  1. test20191210 钟子谦

    100+40+0=140.暴力没写满-- 简单模拟 很久很久以前,有一个 \(1\sim n\) 的排列 \(a\),还有一个长度为 \(q\) 的,每个元素在 \(1\) 到 \(n\) 之间的序列 ...

  2. [Contest20171102]简单数据结构题

    给一棵$n$个点的数,点权开始为$0$,有$q$次操作,每次操作选择一个点,把周围一圈点点权$+1$,在该操作后你需要输出当前周围一圈点点权的异或和. 由于输出量较大,设第$i$个询问输出为$ans_ ...

  3. ioi2018集训队自选题:最短路练习题

    题意:链接 定义pos[i]表示i这个值在数组里的下标. 我们先用单调栈找到每个元素左边和右边第一个比它大的元素$l_i$和$r_i$,然后建一棵二叉树,我们就叫做maxtree吧 (upd:mdzz ...

  4. QDUOJ 一道简单的数据结构题 栈的使用(括号配对)

    一道简单的数据结构题 发布时间: 2017年6月3日 18:46   最后更新: 2017年6月3日 18:51   时间限制: 1000ms   内存限制: 128M 描述 如果插入“+”和“1”到 ...

  5. [LOJ#2326]「清华集训 2017」简单数据结构

    [LOJ#2326]「清华集训 2017」简单数据结构 试题描述 参加完IOI2018之后就是姚班面试.而你,由于讨厌物理.并且想成为乔布斯一样的创业家,被成功踢回贵系. 转眼,时间的指针被指向201 ...

  6. 2019浙大校赛--A--Thanks, TuSimple!(简单模拟题)

    这题前三段都是一堆吹爆赞助商的屁话,正式题目在图片下边,一个简单模拟题. 题目大意: 有n个男生,m个女生在进行舞会,其中一部分男生祥和比自己矮的女生跳舞,一部分男生想和比自己高的女生跳舞,一部分女生 ...

  7. 【Luogu3676】小清新数据结构题(动态点分治)

    [Luogu3676]小清新数据结构题(动态点分治) 题面 洛谷 题解 先扯远点,这题我第一次看的时候觉得是一个树链剖分+线段树维护. 做法大概是这样: 我们先以任意一个点为根,把当前点看成是一棵有根 ...

  8. python学习总结----简单数据结构

    mini-web服务器 - 能够完成简单的请求处理 - 使用http协议 - 目的:加深对网络编程的认识.为后面阶段学习web做铺垫 简单数据结构 - 排列组合 import itertools # ...

  9. 牛客练习赛22-E.简单数据结构1(扩展欧拉定理降幂 +树状数组)

    链接:E.简单数据结构1 题意: 给一个长为n的序列,m次操作,每次操作: 1.区间加 2.对于区间,查询 ,一直到- 请注意每次的模数不同.   题解:扩展欧拉定理降幂 对一个数p取log(p)次的 ...

随机推荐

  1. Linux运维工程师是什么鬼?

    第一部分:定义 运维工程师,字面理解运行维护. linux运维即linux运维工程师,集合网络.系统.数据库.开发.安全工作于一身的“复合性人才”.   除了传统IT运维部分,运维人员还是管理制度.规 ...

  2. NET_Framework_4.0installer.rar

    部署提示: 1.首先下载有关的安装程序 NET_Framework_4.0installer.rar 这是我整理好的四个软件(大致一共10MB),分别如下 WindowsInstaller-KB893 ...

  3. CUDA:Supercomputing for the Masses (用于大量数据的超级计算)-第九节

    原文链接 第九节:使用CUDA拓展高等级语言 Rob Farber 是西北太平洋国家实验室(Pacific Northwest National Laboratory)的高级科研人员.他在多个国家级的 ...

  4. 【思维题 最大权闭合子图】loj#6045. 「雅礼集训 2017 Day8」价

    又是经典模型的好题目 题目描述 人类智慧之神 zhangzj 最近有点胖,所以要减肥,他买了 NN 种减肥药,发现每种减肥药使用了若干种药材,总共正好有 NN 种不同的药材. 经过他的人脑实验,他发现 ...

  5. js原生实现三级联动下拉菜单

    js代码: <!doctype html> <html> <head> <meta charset="utf-8"> <tit ...

  6. django+xadmin在线教育平台(三)

    通过留言版功能回顾django基础知识 将对于django目录结构,使用Django快速搭建可以提交的表单页面,models.py , urls.py, views.py. 从数据库中取出数据展示到h ...

  7. Linux常用关机重启命令

    # shutdown -h 10       //计算机将在10分钟后 关机,且会显示在登录用户的当前屏幕中 # shutdown -h now    //立即 关机 # shutdown -h 20 ...

  8. The Apache Tomcat Servlet/JSP Container

    1.Tomcat部署的场景分析 通常,我们对tomcat单机部署需求可以分为几种: 单实例单应用 (一个tomcat 一个web应用) 单实例多应用 (一个tomcat多个应用) 多实例单应用 (多个 ...

  9. 常见react面试题汇总

    已经开源 地址:https://github.com/nanhupatar...关注我们团队: React 中 keys 的作用是什么? Keys 是 React 用于追踪哪些列表中元素被修改.被添加 ...

  10. 【CodeBase】通过层级键在多维数组中获取目标值

    通过层级键在多维数组中获取目标值 /* *Author : @YunGaZeon *Date : 2017.08.09 *param data : Data Array *param keys : K ...