zoukankan      html  css  js  c++  java
  • NOIP模拟赛 行走

    题目描述

    “我有个愿望,我希望走到你身边。”

    这是个奇异的世界,世界上的n-1条路联结起来形成一棵树,每条路有一个对应的权值ci

    现在我会给出q组询问或操作。

    每次询问我会从一个x点走到y点,初始在x点我会有一个数字v,然后每走过一条权值为c的边,我的v就会变成[v/c],问最后到y时v变成了什么。

    每次修改我会修改一条边的权值,保证修改后的权值小于等于原来的权值且不会小于1。

    每组询问或操作的格式如下:

    询问:1 x y v表示从x走到y,一开始的数字为v。

    操作:2 p c表示将第p条边的边权修改为c

    输入

    第一行两个整数n和q表示点个数和询问与操作个数

    接下来n-1行每行三个整数u,v,c表示u与v之间有一条边权为c的边

    接下来q行每行第一个数type

    如果type=1那么接下来三个数x,y,v表示一组询问

    如果type=2那么接下来两个数p,c表示一组操作

    输出

    对于每组询问输出一个数表示最后的答案

    样例输入1

    6 6
    1 2 1
    1 3 7
    1 4 4
    2 5 5
    2 6 2
    1 4 6 17
    2 3 2
    1 4 6 17
    1 5 5 20
    2 4 1
    1 5 1 3

    样例输出1

    2
    4
    20
    3

    样例输入2

    5 4
    1 2 7
    1 3 3
    3 4 2
    3 5 5
    1 4 2 100
    1 5 4 1
    2 2 2
    1 1 3 4

    样例输出2

    2
    0
    2

    数据范围

    对于70%的数据保证1<=n<=1000

    对于100%的数据保证1<=n<=100000,1<=ci<=10^18

    保证每次修改后的边权小于等于原来的边权且不会小于1

    题解:

    由于对于百分之百的数据,ci可能会出现10^18。

    而对于题目中[v/c]的操作,有几个可以利用的性质:

    1)出现一段连续的边c=1时,不需要操作

    2)当边数超过62时,答案必定为0

        证明2):对∨x∈[1,+∞),x^62>10^18

    因此,对于连续的边出现c=1时,用并查集将这些边集的子集合并,方便操作;对于2)还可以判断当边数>62时,直接输出'0'

    #include<stdio.h>
    #include<string.h>
    #define N 100001
    using namespace std;
    typedef long long ll;
    inline int Fs(){
        int x=0,c=getchar(),f=1;
        for(;c<48||c>57;c=getchar())
            if(!(c^45))
                f=-1;
        for(;c>47&&c<58;c=getchar())
            x=(x<<1)+(x<<3)+c-48;
        return x*f;
    }
    inline ll Fl(){
        ll x=0,c=getchar(),f=1;
        for(;c<48||c>57;c=getchar())
            if(!(c^45))
                f=-1;
        for(;c>47&&c<58;c=getchar())
            x=(x<<1)+(x<<3)+c-48;
        return x*f;
    }
    struct edge{
        ll w;
        int v,nxt;
    }e[N<<1];
    ll s[N],s1[N],s2[N];
    int n,fst[N],fa[N],dep[N],f[N];
    inline void link(int a,int b,ll c){
        static int tt;
        e[++tt].w=c,
        e[tt].v=b,
        e[tt].nxt=fst[a],
        fst[a]=tt;
    }
    void dfs(int x){
        for(int j=fst[x];j;j=e[j].nxt)
            if(!fa[e[j].v])
                fa[e[j].v]=x,
                s[e[j].v]=e[j].w,
                dep[e[j].v]=dep[x]+1,
                dfs(e[j].v);
    }
    int bin(int x){
        int p1,p2=x;
        for(;x^f[x];x=f[x])
            ;
        for(;p2^f[p2];)
            p1=f[p2],
            f[p2]=x,
            p2=p1;
        return x;
    }
    int main(){
        freopen("walk.in","r",stdin),
        freopen("walk.out","w",stdout),
        n=Fs();
        int q=Fs();
        for(int i=1;i<n;i++){
            int x=Fs(),y=Fs();
            ll z=Fl();
            link(x,y,z),
            link(y,x,z);
        }
        dfs(fa[1]=1);
        for(int i=1;i<=n;i++)
            f[i]=i;
        for(int i=1;i<=n;i++)
            if(!(s[i]^1)){
                int y=bin(i);
                int x=bin(fa[i]);
                f[y]=x;
            }
        for(;q;q--){
            ll v;
            int x=Fs(),y,d;
            if(x&1){
                x=Fs();
                y=Fs();
                x=bin(x);
                y=bin(y);
                v=Fl();
                int l1=0,l2=0;
                for(;x^y&&l1+l2<62;)
                    dep[x]>dep[y]?
                    (s1[++l1]=s[x],
                    x=bin(fa[x])):
                    (s2[++l2]=s[y],
                    y=bin(fa[y]));
                if(l1+l2>=62)
                    puts("0");
                else{
                    for(int i=1;i<=l1;i++)
                        v/=s1[i];
                    for(int i=l2;i>=1;i--)
                        v/=s2[i];
                    printf("%I64d
    ",v);
                }
            }
            else{
                d=Fs()<<1;
                v=Fl();
                y=e[d].v;
                x=fa[y];
                fa[e[d-1].v]^y
                ?1:(d--,
                y=e[d].v,
                x=fa[y]);
                s[y]=v;
                if(!(s[y]^1))
                    x=bin(x),
                    y=bin(y),
                    f[y]=x;
            }
        }
        fclose(stdin),
        fclose(stdout);
        return 0;
    }
  • 相关阅读:
    软件补丁问题(网络流24题)
    飞行员配对方案问题(网络流24题)
    [NOIp普及组2011]瑞士轮
    如何在Linux上安装QQ
    [洛谷P2420] 让我们异或吧
    高斯消元
    [SCOI2014]方伯伯的OJ
    [USACO1.3]虫洞
    KMP算法讲解
    [洛谷P1382] 楼房
  • 原文地址:https://www.cnblogs.com/keshuqi/p/6064925.html
Copyright © 2011-2022 走看看