https://vjudge.net/problem/CodeChef-DGCD

https://www.codechef.com/problems/DGCD

题目大意:

给一颗带点权的树,两个操作:

1.将两点间最短路上的点权+d

2.查询两点间最短路上的点权的GCD

显然又是树链剖分,点这里看树链剖分原理

但是我们发现一个问题,我们虽然可以建立线段树维护GCD,但是没有办法处理区间修改问题。

我们考虑更相减损之术的原理,两数做差后的结果和小数的GCD=原来的GCD。

所以我们在维护单点权值的同时维护相邻点权的差值,则GCD(区间内所有相邻点权差的GCD,区间首位点权)就是我们要查的值。

虽然这么说很简单,但是有很多具体细节,大体比较难解决的比如:

1.修改区间的时候,单点权值要用lazy标记维护,而相邻点权的差值就单点修改两次(注意:有些情况下也可能只有一次)即可。

2.询问的时候,单点权值单点查询即可,相邻点权的差值区间查询,注意区间长度为点数-1,也就是说我们可能会碰到空区间,特判掉。

其余具体操作请看代码。

#include<cstdio>
#include<iostream>
using namespace std;
const int N=;
const int INF=;
/*============================**
*************基本操作************
**============================*/
inline int read(){
int X=,w=;char ch=;
while(ch<''||ch>''){w|=ch=='-';ch=getchar();}
while(ch>=''&&ch<='')X=(X<<)+(X<<)+(ch^),ch=getchar();
return w?-X:X;
}
struct node{
int to;
int nxt;
}edge[*N];
struct tree{
int lazy;
int d;
int v;
}t[*N];
int head[N],cnt=,n;
inline void add(int u,int v){
cnt++;
edge[cnt].to=v;
edge[cnt].nxt=head[u];
head[u]=cnt;
return;
}
inline int abs(int x){
return x>?x:-x;
}
int gcd(int x,int y){
return y?gcd(y,x%y):abs(x);
}
int fa[N],dep[N],size[N],son[N],top[N],pos[N],idx[N];
int val[N];
/*============================**
*************树链剖分************
**============================*/
void dfs1(int u){
size[u]=;
for(int i=head[u];i;i=edge[i].nxt){
   int v=edge[i].to;
   if(v==fa[u])continue;
   fa[v]=u;dep[v]=dep[u]+;
   dfs1(v);
     size[u]+=size[v];
   if(!son[u]||size[v]>size[son[u]])son[u]=v;
}
return;
}
int tot;
void dfs2(int u,int anc){
   tot++;
   pos[u]=tot;
   idx[tot]=u;
   top[u]=anc;
   if(!son[u])return;
   dfs2(son[u],anc);
   for(int i=head[u];i;i=edge[i].nxt){
   int v=edge[i].to;
   if(v==fa[u]||v==son[u])continue;
   dfs2(v,v);
}
return;
}
inline void init(){
dfs1();
top[]=idx[]=pos[]=;
tot=;
dfs2(,);
return;
}
/*============================**
************传递lazy************
**============================*/
inline void pushdown(int a,bool is_leaf){
if(is_leaf){
   t[a].v+=t[a].lazy;
}else{
   t[a*].lazy+=t[a].lazy;
   t[a*+].lazy+=t[a].lazy;
}
t[a].lazy=;
return;
}
/*============================**
*************建树操作************
**============================*/
void build(int a,int l,int r){
if(l==r){
   t[a].v=val[idx[l]];
   t[a].d=val[idx[l]]-val[idx[l-]];
   return;
}
int mid=(l+r)>>;
build(a*,l,mid);
build(a*+,mid+,r);
t[a].d=gcd(t[a*].d,t[a*+].d);
return;
}
/*============================**
*************查询操作************
**============================*/
int point_query(int a,int l,int r,int k){
pushdown(a,(l==r));
if(l==r){
   return t[a].v;
}
int mid=(l+r)>>;
if(k<=mid)return point_query(a*,l,mid,k);
return point_query(a*+,mid+,r,k);
}
int range_query(int a,int l,int r,int l1,int r1){
if(l1>r1)return ;
if(r<l1||r1<l)return ;
if(l1<=l&&r<=r1){
   return t[a].d;
}
int mid=(l+r)>>;
return gcd(range_query(a*,l,mid,l1,r1),range_query(a*+,mid+,r,l1,r1));
}
int path_query(int u,int v){
if(top[u]!=top[v]){
   if(dep[top[u]]<dep[top[v]]){int t=u;u=v;v=t;}
   if(top[u]!=u)
   return gcd(path_query(fa[top[u]],v),gcd(range_query(,,n,pos[son[top[u]]],pos[u]),point_query(,,n,pos[top[u]])));
   return gcd(path_query(fa[top[u]],v),point_query(,,n,pos[top[u]]));
}
if(dep[u]>dep[v]){int t=u;u=v;v=t;}
if(u!=v)
   return gcd(point_query(,,n,pos[u]),range_query(,,n,pos[son[u]],pos[v]));
return point_query(,,n,pos[u]);
}
/*============================**
*************修改操作************
**===========================**/
void point_modi(int a,int l,int r,int k,int c){
if(l==r){
   t[a].d+=c;
   return;
}
int mid=(l+r)>>;
if(k<=mid)point_modi(a*,l,mid,k,c);
else point_modi(a*+,mid+,r,k,c);
t[a].d=gcd(t[a*].d,t[a*+].d);
return;
}
void range_modi(int a,int l,int r,int l1,int r1,int v){
if(r1<l||r<l1)return;
pushdown(a,(l==r));
if(l1<=l&&r<=r1){
   t[a].lazy+=v;
   return;
}
int mid=(l+r)>>;
range_modi(a*,l,mid,l1,r1,v);
range_modi(a*+,mid+,r,l1,r1,v);
return;
}
void path_modi(int u,int v,int c){
if(top[u]!=top[v]){
  if(dep[top[u]]<dep[top[v]]){int t=u;u=v;v=t;}
  point_modi(,,n,pos[top[u]],c);
   if(son[u]!=u)point_modi(,,n,pos[son[u]],-c);
   range_modi(,,n,pos[top[u]],pos[u],c);
   path_modi(fa[top[u]],v,c);
   return;
}
if(dep[u]>dep[v]){int t=u;u=v;v=t;}
point_modi(,,n,pos[u],c);
if(son[v])point_modi(,,n,pos[son[v]],-c);
range_modi(,,n,pos[u],pos[v],c);
return;
}
/*============================**
*************主程序段************
**============================*/
int main(){
n=read();
for(int i=;i<=n;i++){
   int u=read()+;
  int v=read()+;
  add(u,v);
   add(v,u);
}
for(int i=;i<=n;i++)val[i]=read();
init();
build(,,n);
int q=read();
while(q--){
char op=;
while(op!='F'&&op!='C')op=getchar();
  if(op=='C'){
   int a=read()+;
   int b=read()+;
   int c=read();
   path_modi(a,b,c);
  }else{
   int a=read()+;
   int b=read()+;
   printf("%d\n",path_query(a,b));
   }
}
return ;
}

CC DGCD:Dynamic GCD——题解的更多相关文章

  1. CodeChef DGCD Dynamic GCD

    CodeChef题面 Time limit 210 ms Code length Limit //内存限制也不说一下,真是的-- 50000 B OS Linux Language limit C, ...

  2. codechef Dynamic GCD [树链剖分 gcd]

    Dynamic GCD 题意:一棵树,字词树链加,树链gcd 根据\(gcd(a,b)=gcd(a,a-b)\) 得到\(gcd(a_1, a_2, ..., a_i) = gcd(a_1, a_1- ...

  3. HDU5726:GCD——题解

    题目:hdu的5726 (我原博客的东西,正好整理过来,属于st表裸题) (可以看出我当时有多么的菜--) 这道题写了一遍,然而蒟蒻的我的时间爆炸了-- 于是看了一下学长的代码(顺便在此处%一下学长) ...

  4. 洛谷 P2568 GCD 题解

    原题链接 庆祝一下:数论紫题达成成就! 第一道数论紫题.写个题解庆祝一下吧. 简要题意:求 \[\sum_{i=1}^n \sum_{j=1}^n [gcd(i,j)==p] \] 其中 \(p\) ...

  5. CodeChef Dynamic GCD

    嘟嘟嘟vjudge 我今天解决了一个历史遗留问题! 题意:给一棵树,写一个东西,支持一下两种操作: 1.\(x\)到\(y\)的路径上的每一个点的权值加\(d\). 2.求\(x\)到\(y\)路径上 ...

  6. BZOJ2820:YY的GCD——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2820 Description 神犇YY虐完数论后给傻×kAc出了一题给定N, M,求1<=x& ...

  7. BZOJ1901:Zju2112 Dynamic Rankings——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=1901 Description 给定一个含有n个数的序列a[1],a[2],a[3]……a[n],程序 ...

  8. Dynamic Gcd

    树链剖分+差分 直接区间加显然是不行的,由于gcd(a,b,c)=gcd(a,a-b,b-c),那么我们对这些数差分,然后就变成单点修改.原本以为这道题很简单,没想到这么麻烦,就膜了发代码. 首先我们 ...

  9. CH 4302 Interval GCD 题解

    题意 给定一个长度为N的数列A,以及M条指令 (N≤5* 10^5, M<=10^5),每条指令可能是以下两种之一: "C l r d",表示把 A[l],A[l+1],-, ...

随机推荐

  1. 用PowerDesign反向生成数据库Sql语句问题

            在用Pd15反向生成数据库时,生成的Sql语句在Sql Server Manager Studio里面报错,根本就执行不了.数据库用的是Sql Server 2008 R2.经过一番修 ...

  2. JDK1.8改为JDK1.7过程

    电脑之前eclipse版本要求JDK1.8版本,现在要用jboss7.1做性能测试,目前仅支持JDK7.故需要降级. 网上有很多说把1.8删掉,这种做法我是不建议的,那么要用的时候呢?又得装回来多蛋疼 ...

  3. 初识c++模板元编程

    模板元编程(Template metaprogramming,简称TMP)是编译器内执行的程序,编译器读入template,编译输出的结果再与其他源码一起经过普通编译过程生成目标文件.通俗来说,普通运 ...

  4. Qt 解析EXcel文件

    写代码需要将excel中的文件导入到数据库中 网上找到以为大神写的,但是当初没有保存,也没有找到 我几乎是原分不动拔下来的,希望大神莫怪 void AddDialog::readExcel(QStri ...

  5. Jquery获取DOM绑定事件

    获取到当前正在执行的事件: $('#testDive').bind('click', function(event){alert('event: ' + event.type)}); 获取所有绑定事件 ...

  6. [USACO09Open] Tower of Hay 干草塔

    为了调整电灯亮度,贝西要用干草包堆出一座塔,然后爬到牛棚顶去把灯泡换掉.干草包会从传送带上运来,共会出现N包干草,第i包干草的宽度是W i ,高度和长度统一为1.干草塔要从底层开始铺建.贝西会选择最先 ...

  7. token接口的测法

    接口一般都有权限的校验,一般是需要登录后才可以调用 对于接口的认证,一般通过两种方式来实现1.校验用户请求中是否包含某项指定的cookie2.校验用户的请求的header中是否包含某项指定的字段(to ...

  8. Java基础知识:Java实现Map集合二级联动3

    * Returns an image stored in the file at the specified path * @param path String The path to the ima ...

  9. nginx web服务器的安装使用

    nginx是一个web服务器(高性能web服务器),类似于apache服务器和iis服务器,由于nginx服务器具有轻量级高并发的特点,目前nginx的使用已经超越了apache. nginx介绍:n ...

  10. 1.linux环境配置

    首先说一下,这里是虚拟机环境. 1.用vbox安装centos6.8-mini 注意不要使用复制的方式安装,复制的虚拟机网络不通 安装如下: 主机 ip 角色 内存 hadoop1 192.168.0 ...