BZOJ

表示非常爽2333


\(Description\)

给定一棵\(n\)个点的树,每个点有一个属性\(1\leq r_i\leq R\)。

\(Q\)次询问,每次询问给定\(r1,r2\),求有多少点对\((x,y)\)满足,\(r_x=r1,\ r_y=r2\),且\(x\)是\(y\)的祖先。

\(n,q\leq2\times10^5,\ R\leq25000\)。

\(Solution\)

对属性为\(r2\)的有多少个点分类讨论。

若\(\leq\sqrt n\),在每个点处暴力统计它祖先的贡献;

若\(\gt\sqrt n\),则这样的属性不超过\(\sqrt n\)种,在属性为\(r1\)的点上暴力枚举这些\(r2\)更新答案(这里可以差分:进入子树前与访问完子树后)。

显然对于第二种情况,在每个点上,要对询问的\(r2\)去重才能保证复杂度(然而数据没卡不去重好像也能过...)。

关于如何去重,自己想的是,对\(r2\)相同的一些询问,要同时更新它们。大概以\(r2\)为关键字sort下,就可以差分了?

事实上对询问点对\((r1,r2)\)判一下重就可以了...如果出现够询问\((r1,r2)\),就直接用之前的作答案。

复杂度\(O(n\sqrt n+q\sqrt n)\)。

树分块做法,然而懒得看。


//28360kb	7720ms
#include <map>
#include <cmath>
#include <cstdio>
#include <cctype>
#include <vector>
#include <algorithm>
#define mp std::make_pair
#define pr std::pair<int,int>
#define gc() getchar()
#define MAXIN 500000
//#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
typedef long long LL;
const int N=2e5+5,M=25005; int Enum,H[N],nxt[N],A[N],Ans[N];
std::vector<pr> v1[M],v2[M];
char IN[MAXIN],*SS=IN,*TT=IN; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
inline void AE(int u,int v)
{
nxt[v]=H[u], H[u]=v;
}
void DFS(int x)
{
static int sum1[N],sum2[N]; const std::vector<pr> &vec2=v2[A[x]];//一开始&忘了写...我说怎么MLE=-=
for(int i=0,l=vec2.size(); i<l; ++i)
Ans[vec2[i].second]+=sum2[vec2[i].first]; ++sum1[A[x]], ++sum2[A[x]];
const std::vector<pr> &vec1=v1[A[x]];
for(int i=0,l=vec1.size(); i<l; ++i)
Ans[vec1[i].second]-=sum1[vec1[i].first]; for(int v=H[x]; v; v=nxt[v]) DFS(v); for(int i=0,l=vec1.size(); i<l; ++i)
Ans[vec1[i].second]+=sum1[vec1[i].first];
--sum2[A[x]];
} int main()
{
static int pos[N],cnt[M]; const int n=read(),R=read(),Q=read(),size=sqrt(n);
++cnt[A[1]=read()];
for(int i=2; i<=n; ++i) AE(read(),i), ++cnt[A[i]=read()]; std::map<pr,int> f;
std::map<pr,int>::iterator it;
for(int i=1; i<=Q; ++i)
{
int r1=read(),r2=read();
if((it=f.find(mp(r1,r2)))==f.end())
{
f[mp(r1,r2)]=pos[i]=i;
if(cnt[r2]<=size) v2[r2].push_back(mp(r1,i));
else v1[r1].push_back(mp(r2,i));
}
else pos[i]=it->second;
}
DFS(1);
for(int i=1; i<=Q; ++i) printf("%d\n",Ans[pos[i]]);//其实应该用long long的=v= return 0;
}

BZOJ.3351.[IOI2009]Regions(根号分治 差分)的更多相关文章

  1. BZOJ3351: [ioi2009]Regions(根号分治)

    题意 题目链接 Sol 很神仙的题 我们考虑询问(a, b)(a是b的祖先),直接对b根号分治 如果b的出现次数\(< \sqrt{n}\),我们可以直接对每个b记录下与它有关的询问,这样每个询 ...

  2. BZOJ 3351: [ioi2009]Regions

    对于一个询问(x,y)对y出现次数分类,若<=lim,在儿子处统计答案,若>lim则y的种类肯定<lim,在祖先处统计(仿佛要去重?但是没去重也过了,那个时限仿佛怎么做都能过) #i ...

  3. BZOJ.4320.[ShangHai2006]Homework(根号分治 分块)

    BZOJ \(\mathbb{mod}\)一个数\(y\)的最小值,可以考虑枚举剩余系,也就是枚举区间\([0,y),[y,2y),[2y,3y)...\)中的最小值(求后缀最小值也一样)更新答案,复 ...

  4. [CF587F]Duff is Mad[AC自动机+根号分治+分块]

    题意 给你 \(n\) 个串 \(s_{1\cdots n}\) ,每次询问给出 \(l,r,k\) ,问在 \(s_{l\cdots r}\) 中出现了多少次 \(s_k\) . \(n,q,\su ...

  5. CF587F-Duff is Mad【AC自动机,根号分治】

    正题 题目链接:https://www.luogu.com.cn/problem/CF587F 题目大意 给出\(n\)个字符串\(s\).\(q\)次询问给出\(l,r,k\)要求输出\(s_{l. ...

  6. BZOJ.4184.shallot(线段树分治 线性基)

    BZOJ 裸的线段树分治+线性基,就是跑的巨慢_(:з」∠)_ . 不知道他们都写的什么=-= //41652kb 11920ms #include <map> #include < ...

  7. Codeforces 1039D You Are Given a Tree [根号分治,整体二分,贪心]

    洛谷 Codeforces 根号分治真是妙啊. 思路 考虑对于单独的一个\(k\)如何计算答案. 与"赛道修建"非常相似,但那题要求边,这题要求点,所以更加简单. 在每一个点贪心地 ...

  8. CF1039E Summer Oenothera Exhibition 贪心、根号分治、倍增、ST表

    传送门 感谢这一篇博客的指导(Orzwxh) $PS$:默认数组下标为$1$到$N$ 首先很明显的贪心:每一次都选择尽可能长的区间 不妨设$d_i$表示在取当前$K$的情况下,左端点为$i$的所有满足 ...

  9. CF1039E Summer Oenothera Exhibition 根号分治,LCT,ST表

    CF1039E Summer Oenothera Exhibition LG传送门 根号分治好题. 可以先看我的根号分治总结. 题意就是给出长度为\(n\)的区间和\(q\)组询问以及一个\(w\), ...

随机推荐

  1. cf1110F 离线+树上操作+线段树区间更新

    自己搞的算法超时了..但是思路没什么问题:用线段树维护每个点到叶子节点的距离即可 /* 线段树维护区间最小值,每次向下访问,就把访问到的点对应的区间段减去边权 到另一颗子树访问时,向上回溯时加上减去的 ...

  2. Python模块的导入以及软件开发规范

    Python文件的两种用途 1 . 当脚本直接使用,直接当脚本运行调用即可 def func(): print("from func1") func() 2 . 当做模块被导入使用 ...

  3. webpack学习笔记--其它配置项

     其它配置项 除了前面介绍到的配置项外,Webpack 还提供了一些零散的配置项.下面来介绍它们中常用的部分. Target JavaScript 的应用场景越来越多,从浏览器到 Node.js,这些 ...

  4. [转] babel入门基础

    背景 babel的官网说babel是下一代的js语法编译器,现在自己也在很多项目中使用了babel,可是自己对babel的认识呢,只停留在从google和别人项目中copy的配置代码上,内心感到很不安 ...

  5. es6 新增数据类型Symbol

    es6在string number boolean null undefined object之外又新增了一种Symbol类型. Symbol意思是符号,有一个特性—每次创建一个Symbol值都是不一 ...

  6. 使用jquery获取父元素或父节点

    使用jquery获取父元素或父节点,比较简单,jquery提供了丰富的方法来让我们使用jquery获取父元素或父节点   jquery获取父元素方法比较多,比如parent(),parents(),c ...

  7. 【BZOJ3782】上学路线

    题解: 这个容斥以前做过 到i点的所有方案显然是个组合数 然后要减去不合法的方案数 我们可以考虑成减去到每个障碍点为第一次遇到的障碍然后之后乱走就可以了 因为模数不是质数,所以crt合并

  8. C# 之 比较两个word文档的内容

    利用 Microsoft.Office.Interop.Word 组件进行比较.引入命名空间:using Word2013 = Microsoft.Office.Interop.Word; 代码如下: ...

  9. 【Android】ContentValues的用法

    ContentValues 和HashTable类似都是一种存储的机制 但是两者最大的区别就在于,contenvalues只能存储基本类型的数据,像string,int之类的,不能存储对象这种东西,而 ...

  10. Telnet和SSH

    1. 协议用途 Telnet和SSH用于远程访问服务器的的两大常用协议.利用它们,我们可以管理并监控生产服务器和企业服务器,更新服务器内核,安装最新的软件包和补丁,能够远程登录服务器,开展软件开发.测 ...