bzoj3251 树上三角形

Description

给定一大小为n的有点权树,每次询问一对点(u,v),问是否能在u到v的简单路径上取三个点权,以这三个权值为边长构成一个三角形。同时还支持单点修改。

Input

第一行两个整数n、q表示树的点数和操作数
第二行n个整数表示n个点的点权
以下n-1行,每行2个整数a、b,表示a是b的父亲(以1为根的情况下)
以下q行,每行3个整数t、a、b
若t=0,则询问(a,b)
若t=1,则将点a的点权修改为b

Output

对每个询问输出一行表示答案,“Y”表示有解,“N”表示无解。

Sample Input

5 5
1 2 3 4 5
1 2
2 3
3 4
1 5
0 1 3
0 4 5
1 1 4
0 2 5
0 2 3

Sample Output

N
Y
Y
N

HINT

对于100%的数据,n,q<=100000,点权范围[1,2^31-1]

Key To Problem

这道题实在脑洞好大,如果建树判断一条树上是否能构成三角形,那显然是不行的,所以要先判断在一定的数中最多有哪些数可以构成三角形,那一定就是斐波那契数列,int范围内的斐波那契数一共是46个,所以只要一串数超过46个就一定可以构成三角形,如果不到46个,那就将这串数排序查询,之后就是裸的LCA

Code

#include<vector>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 100010
#define M 46
using namespace std;
typedef long long ll;
int n,m,tot;
int a[N];
int head[N];
int tail[N];
int next[N];
int dfn[N];
int fa[N];
int Q[50];

int cmp(const void *x,const void *y)
{
    return *(int *)x-*(int *)y;
}

void add(int x,int y){next[++tot]=head[x];head[x]=tot;tail[tot]=y;}

void dfs(int u,int deep)
{
    dfn[u]=deep;
    for(int i=head[u];i;i=next[i])
    {
        int to=tail[i];
        dfs(to,deep+1);
    }
}

bool LCA(int x,int y)
{
    int top=0;
    if(dfn[x]<dfn[y])
        swap(x,y);
    while(top<=M&&dfn[x]>dfn[y])
        Q[++top]=a[x],x=fa[x];
    if(top>M)
        return true;
    while(top<=M&&x!=y)
        Q[++top]=a[x],Q[++top]=a[y],x=fa[x],y=fa[y];
    Q[++top]=a[x];
    if(top>M)
        return true;
    qsort(Q+1,top,sizeof(Q[0]),cmp);
    for(int i=3;i<=top;i++)
    {
        if(Q[i]<(ll)Q[i-1]+Q[i-2])
            return true;
    }
    return false;
}

int main()
{
//  freopen("tree.in","r",stdin);
//  freopen("tree.out","w",stdout);
    cin>>n>>m;
    for(int i=1;i<=n;i++)
        scanf("%d",&a[i]);
    for(int i=1;i<n;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        fa[y]=x;
        add(x,y);
    }
    dfs(1,1);
    for(int i=1;i<=m;i++)
    {
        int x,y,z;
        scanf("%d%d%d",&x,&y,&z);
        if(x==1) a[y]=z;
        else
        {
            if(LCA(y,z)) puts("Y");
            else puts("N");
        }
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值