给一颗n个点的有点权的树,有m个询问,对于每个询问u,v,k,首先将点u到点v的最短路径上的所有点按顺序编号,u的编号为1,求树链上所有点的新编号cnt满足cnt%k==0的点的权值的最大值。n,m,k<=10^5

根据k的大小分成两部分处理。原问题可转化为 deep[i] % k = a / b 。

对于k较大的,直接暴力,按照dfs序用一个栈记录下所经过的点,对于每个询问的点不停往上爬。

对于k较小的,将询问按照k分类。对于每一种k,将所有点按照dep[i] % k分类,将每个点树链剖分后hash下来的坐标再按照dep[i] % k映射到一起,用线段树进行维护。

每次查询deep[i] % k = a 时,相当于在某个区间查询最大值。

#include <map>
#include <set>
#include <stack>
#include <queue>
#include <cmath>
#include <ctime>
#include <string>
#include <vector>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cassert>
#include <iostream>
#include <algorithm>
#pragma comment(linker,"/STACK:102400000,102400000")
using namespace std; #define N 100008
#define LL long long
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define clr(x,v) memset(x,v,sizeof(x));
#define bitcnt(x) __builtin_popcount(x)
#define rep(x,y,z) for (int x=y;x<=z;x++)
#define repd(x,y,z) for (int x=y;x>=z;x--)
const int mo = ;
const int inf = 0x3f3f3f3f;
const int INF = ;
/**************************************************************************/ int T,n,m,k,sum,block,stop,cq2,tmp,label;
int lt[N],a[N],dep[N],st[N],ans[N],size[N],son[N],rk[N],w[N],top[N],mx[N<<],cl[N],cr[N],fuck[N];;
int f[N][];
struct line{
int u,v,nt;
}eg[N*];
struct que{
int u,v,lca,k,id,flag;
};
vector <que> q1[N],q2[N];
void add(int u,int v){
eg[++sum]=(line){u,v,lt[u]};
lt[u]=sum;
}
void init(){
clr(lt,); clr(ans,);
sum=; tmp=;
}
void dfs(int u,int fa){
f[u][]=fa; dep[u]=dep[fa]+; size[u]=; son[u]=;
for (int i=lt[u];i;i=eg[i].nt){
int v=eg[i].v;
if (v==fa) continue;
dfs(v,u);
size[u]+=size[v];
if (size[v]>size[son[u]]) son[u]=v;
}
}
void dfs_2(int u,int tp){
top[u]=tp; w[u]=++tmp; rk[tmp]=u;
if (son[u]) dfs_2(son[u],tp);
for (int i=lt[u];i;i=eg[i].nt){
int v=eg[i].v;
if (v==f[u][] || v==son[u]) continue;
dfs_2(v,v);
}
}
int lca(int x,int y){
if (dep[x]<dep[y]) swap(x,y);
int d=dep[x]-dep[y];
repd(i,,)
if (d & (<<i)) x=f[x][i];
if (x==y) return x;
repd(i,,)
if (f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i];
return f[x][];
}
void work_1(int u){
st[++stop]=u;
for (int i=;i<q1[u].size();i++){
int v=q1[u][i].v,x=q1[u][i].lca,k=q1[u][i].k,id=q1[u][i].id,flag=q1[u][i].flag;
if (flag)
{
int tmp=stop-k+;
while (tmp>= && dep[st[tmp]]>=dep[x]){
ans[id]=max(ans[id],a[st[tmp]]);
tmp-=k;
}
}
else
{
int tmp=dep[x]+ k - (dep[v]-dep[x]+) % k ;
while (tmp<=stop && dep[st[tmp]]<=dep[u]){
ans[id]=max(ans[id],a[st[tmp]]);
tmp+=k;
}
}
}
for (int i=lt[u];i;i=eg[i].nt){
int v=eg[i].v;
if (v==f[u][]) continue;
work_1(v);
}
st[stop--]=;
}
int query(int L,int R,int l,int r,int rt){
if (L<=l && r<=R){
return mx[rt];
}
int m=(l+r)>>;
int res=;
if (L <= m) res=max(res,query(L,R,lson));
if (m < R) res=max(res,query(L,R,rson));
return res;
} void build(int l,int r,int rt){
mx[rt]=;
if (l==r){
mx[rt]=a[rk[fuck[l]]];
return;
}
int m=(l+r)>>;
build(lson);
build(rson);
mx[rt]=max(mx[rt<<],mx[rt<<|]);
}
int calc(int L, int R, int k) {
int l = lower_bound(fuck + cl[k], fuck + cr[k] + , L) - fuck;
int r = upper_bound(fuck + cl[k], fuck + cr[k] + , R) - fuck - ;
return l <= r ? query(l,r,,n,) : ;
}
void find(int tp,int x,int cl,int id){
while (top[x]!=top[tp]){
ans[id]=max(ans[id],calc(w[top[x]],w[x],cl));
x=f[top[x]][];
}
ans[id]=max(ans[id],calc(w[tp],w[x],cl));
} vector <int> E[N];
void work_2(int k){
rep(i,,k) E[i].clear();
rep(i,,n) E[(dep[rk[i]]) % k].push_back(i);
label=;
rep(i,,k-){
cl[i]=label+;
for (int j=;j<E[i].size();j++)
fuck[++label]=E[i][j];
cr[i]=label; }
build(,n,);
for (int i=;i<q2[k].size();i++){
int u=q2[k][i].u,v=q2[k][i].v,x=q2[k][i].lca,id=q2[k][i].id;
int tmp1=(dep[x]+(dep[u]-dep[x]+) % k) % k;
int tmp2=(dep[x]+k-(dep[u]-dep[x]+) % k) % k;
find(x,u,tmp1,id);
find(x,v,tmp2,id);
}
}
int main(){
int cas=;
scanf("%d",&T);
while (T--){
printf("Case #%d:\n",++cas );
init();
scanf("%d%d",&n,&m);
rep(i,,n) scanf("%d",&a[i]);
rep(i,,n-){
int u,v;
scanf("%d%d",&u,&v);
add(u,v); add(v,u);
}
dfs(,);
dfs_2(,);
rep(j,,)
rep(i,,n)
f[i][j]=f[f[i][j-]][j-];
block=;
cq2=;
rep(i,,n) q1[i].clear();
rep(i,,n) q2[i].clear();
rep(i,,m){
int u,v,k;
scanf("%d%d%d",&u,&v,&k);
int x=lca(u,v);
if (k>block)
{
q1[u].push_back((que){u,v,x,k,i,});
q1[v].push_back((que){v,u,x,k,i,});
}
else
{
q2[k].push_back((que){u,v,x,k,i,});
}
}
work_1();
rep(i,,block) if (q2[i].size()) work_2(i);
rep(i,,m) printf("%d\n",ans[i]);
}
}

HDU5840 Problem This world need more Zhu 分块 树剖的更多相关文章

  1. BZOJ4381 : [POI2015]Odwiedziny / Luogu3591[POI2015]ODW - 分块+树剖

    Solution 在步伐$pace$比较小的时候, 我们发现用前缀和直接维护会很快 而在$pace$比较大的时候, 则暴力往上跳会最优 设$blo= \sqrt{N}$ 若$pace<=blo$ ...

  2. HDU5840 (分块+树链剖分)

    Problem This world need more Zhu 题目大意 给一颗n个点的有点权的树,有m个询问,对于每个询问u,v,k,首先将点u到点v的最短路径上的所有点按顺序编号,u的编号为1, ...

  3. 【BZOJ 3295】动态逆序对 - 分块+树状数组

    题目描述 给定一个1~n的序列,然后m次删除元素,每次删除之前询问逆序对的个数. 分析:分块+树状数组 (PS:本题的CDQ分治解法见下一篇) 首先将序列分成T块,每一块开一个树状数组,并且先把最初的 ...

  4. 2018.06.30 BZOJ4765: 普通计算姬(dfs序+分块+树状数组)

    4765: 普通计算姬 Time Limit: 30 Sec Memory Limit: 256 MB Description "奋战三星期,造台计算机".小G响应号召,花了三小时 ...

  5. 【UOJ#435】【集训队作业2018】Simple Tree 分块+树链剖分

    题目大意: 有一棵有根树,根为 1 ,点有点权.现在有 m 次操作,操作有 3 种:1 x y w ,将 x 到 y 的路径上的点点权加上 w (其中 w=±1w=±1 ):2 x y ,询问在 x ...

  6. 【BZOJ4999】This Problem Is Too Simple!(线段树)

    [BZOJ4999]This Problem Is Too Simple!(线段树) 题面 BZOJ 题解 对于每个值,维护一棵线段树就好啦 动态开点,否则空间开不下 剩下的就是很简单的问题啦 当然了 ...

  7. 【bzoj2141】排队 分块+树状数组

    题目描述 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和.红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别, ...

  8. 【bzoj3744】Gty的妹子序列 分块+树状数组+主席树

    题目描述 我早已习惯你不在身边, 人间四月天 寂寞断了弦. 回望身后蓝天, 跟再见说再见…… 某天,蒟蒻Autumn发现了从 Gty的妹子树(bzoj3720) 上掉落下来了许多妹子,他发现 她们排成 ...

  9. 【分块+树状数组】codechef November Challenge 2014 .Chef and Churu

    https://www.codechef.com/problems/FNCS [题意] [思路] 把n个函数分成√n块,预处理出每块中各个点(n个)被块中函数(√n个)覆盖的次数 查询时求前缀和,对于 ...

随机推荐

  1. defineProperty

    ### Object.defineProperty() https://segmentfault.com/a/1190000007434923方法会直接在一个对象上定义一个新属性,或者修改一个已经存在 ...

  2. leetcode 658找到k个最接近的元素

    class Solution { public: vector<int> findClosestElements(vector<int>& arr, int k, in ...

  3. DPI的理解

    DPI(Dots Per Inch,每英寸点数)是一个量度单位,用于点阵数码影像,指每一英寸长度中,取样.可显示或输出点的数目. DPI是打印机.鼠标等设备分辨率的度量单位.是衡量打印机打印精度的主要 ...

  4. committed与urgent的区别

    Committed跟Urgent都是time automaton 中用来表示state的关键词. 它们的主要区别是: Committed前后的两个状态改变(transition)是串行发生,不可打断的 ...

  5. 关于Toad的Cannot load OCI DLL问题

    昨天重新安装了新版本的JDK,突然发现Toad连接的时候报Cannot load OCI DLL....问题,网上查找了多种方法均不见效. 后调整系统环境变量配置,还原了之前安装的JDK版本,问题修复 ...

  6. Linux监控命令之==>lsof

    一.命令说明 lsof 命令的原始功能是列出打开的文件的进程,但LINUX 下,所有的设备都是以文件的行式存在的,所以,lsof 的功能很强大. 二.参数说明 -a :列出打开文件存在的进程 -c&l ...

  7. Windows客户端 Linux服务器通讯 字符编码问题

    Windows下的字符编码默认是gb2312 在Linux下需要转成utf8 才能正确的看到对应的中文编码 提供转换函数 /*------------------------------------- ...

  8. Java基础之Volatile原理

    原文链接: http://www.aoaoyi.com/archives/956.html 计算机在执行程序时,每条指令都是在CPU中执行的,而执行指令过程中,势必涉及到数据 的读取和写入.由于程序运 ...

  9. 车牌识别1:License Plate Detection and Recognition in Unconstrained Scenarios阅读笔记

    一.WHAT 论文下载地址:License Plate Detection and Recognition in Unconstrained Scenarios [pdf] github 的项目地址: ...

  10. JS 截取字符的方法

    substr() 方法 substr() 方法可在字符串中抽取从 start 下标开始的指定数目的字符. stringObject.substr(start,length) start:必需.要抽取的 ...