zoukankan      html  css  js  c++  java
  • [ZJOI2012]网络

    题目描述

    有一个无向图G,每个点有个权值,每条边有一个颜色。这个无向图满足以下两个条件:

    1. 对于任意节点连出去的边中,相同颜色的边不超过两条。

    2. 图中不存在同色的环,同色的环指相同颜色的边构成的环。

    在这个图上,你要支持以下三种操作:

    1. 修改一个节点的权值。

    2. 修改一条边的颜色。

    3. 查询由颜色c的边构成的图中,所有可能在节点u到节点v之间的简单路径上的节点的权值的最大值。

    输入输出格式

    输入格式:

    输入文件network.in的第一行包含四个正整数N, M, C, K,其中N为节点个数,M为边数,C为边的颜色数,K为操作数。

    接下来N行,每行一个正整数vi,为节点i的权值。

    之后M行,每行三个正整数u, v, w,为一条连接节点u和节点v的边,颜色为w。满足1 ≤ u, v ≤ N,0 ≤ w < C,保证u ≠ v,且任意两个节点之间最多存在一条边(无论颜色)。

    最后K行,每行表示一个操作。每行的第一个整数k表示操作类型。

    1. k = 0为修改节点权值操作,之后两个正整数x和y,表示将节点x的权值vx修改为y。

    2. k = 1为修改边的颜色操作,之后三个正整数u, v和w,表示将连接节点u和节点v的边的颜色修改为颜色w。满足0 ≤ w < C。

    3. k = 2为查询操作,之后三个正整数c, u和v,表示查询所有可能在节点u到节点v之间的由颜色c构成的简单路径上的节点的权值的最大值。如果不存在u和v之间不存在由颜色c构成的路径,那么输出“-1”。

    输出格式:

    输出文件network.out包含若干行,每行输出一个对应的信息。

    1. 对于修改节点权值操作,不需要输出信息。

    2. 对于修改边的颜色操作,按以下几类输出:

    a) 若不存在连接节点u和节点v的边,输出“No such edge.”。

    b) 若修改后不满足条件1,不修改边的颜色,并输出“Error 1.”。

    c) 若修改后不满足条件2,不修改边的颜色,并输出“Error 2.”。

    d) 其他情况,成功修改边的颜色,并输出“Success.”。

    输出满足条件的第一条信息即可,即若同时满足b和c,则只需要输出“Error 1.”。

    1. 对于查询操作,直接输出一个整数。

    输入输出样例

    输入样例#1: 复制
    4 5 2 7
    1
    2
    3
    4
    1 2 0
    1 3 1
    2 3 0
    2 4 1
    3 4 0
    2 0 1 4
    1 1 2 1
    1 4 3 1
    2 0 1 4
    1 2 3 1
    0 2 5
    2 1 1 4
    输出样例#1: 复制
    4
    Success.
    Error 2.
    -1
    Error 1.
    5

    说明

    颜色0为实线的边,颜色1为虚线的边,

    由颜色0构成的从节点1到节点4的路径有1 – 2 – 4,故max{v1, v2, v4} = max{ 1, 2, 4 } = 4。

    将连接节点1和节点2的边修改为颜色1,修改成功,输出“Success.”

    将连接节点4和节点3的边修改为颜色1,由于修改后会使得存在由颜色1构成的环( 1 – 2 – 4 – 3 – 1 ),不满足条件2,故不修改,并输出“Error 2”。

    不存在颜色0构成的从节点1到节点4的边,输出“-1”。

    将连接节点2和节点3的边修改为颜色1,由于修改后节点2的连出去的颜色为1的边有3条,故不满足条件1,故不修改,并输出“Error 1.”。

    将节点2的权值修改为5。

    由颜色1构成的从节点1到节点4的路径有 1 – 2 – 4,故max{v1, v2, v4} = max{ 1, 5, 4 } = 5。

    【数据规模】

    对于30%的数据:N ≤ 1000,M ≤ 10000,C ≤ 10,K ≤ 1000。

    另有20%的数据:N ≤ 10000,M ≤ 100000,C = 1,K ≤ 100000。

    对于100%的数据:N ≤ 10000,M ≤ 100000,C ≤ 10,K ≤ 100000。

    题解:

    思维难度比较低,细节很多,首先看到C<=10,果断开10棵LCT。

      1.对于操作1,在每一个LCT中修改这个点的点权就好。

      2.对于操作2,我的方法是用一个链表来存边,如果a和b之间没有边,就输出“No such edge”。用一个数组in来存出度,如果出度>2并且颜色改变,就输出“Error 2.”。然后判断一下这两个点是否联通,如果联通且不是直接联通,就输出“Error 2.”。其他情况直接link就好。

      1 //Never forget why you start
      2 #include<iostream>
      3 #include<cstdio>
      4 #include<cstdlib>
      5 #include<cstring>
      6 #include<cmath>
      7 #include<algorithm>
      8 #define ll(x) lct[x].child[0]
      9 #define rr(x) lct[x].child[1]
     10 #define son(x,t) lct[x].child[t]
     11 using namespace std;
     12 int n,m,C,k;
     13 int in[10005][10];
     14 struct Tree{
     15   struct LCT{
     16     int child[2],fa,size,x,mmax,rev;
     17     bool is_root;
     18   }lct[10005];
     19   void push_up(int x){
     20     lct[x].size=lct[ll(x)].size+lct[rr(x)].size+1;
     21     lct[x].mmax=max(lct[x].x,max(lct[ll(x)].mmax,lct[rr(x)].mmax));
     22   }
     23   void push_rev(int x){
     24     if(!x)return;
     25     swap(ll(x),rr(x));
     26     lct[x].rev^=1;
     27   }
     28   void push_down(int x){
     29     if(lct[x].rev){
     30       push_rev(ll(x));
     31       push_rev(rr(x));
     32       lct[x].rev^=1;
     33     }
     34   }
     35   void push(int x){
     36     if(!lct[x].is_root)push(lct[x].fa);
     37     push_down(x);
     38   }
     39   int getson(int x){
     40     return x==son(lct[x].fa,1);
     41   }
     42   void rotate(int x){
     43     if(lct[x].is_root)return;
     44     int fa=lct[x].fa,fafa=lct[fa].fa,t=getson(x);
     45     son(fa,t)=son(x,!t);if(son(x,!t))lct[son(x,!t)].fa=fa;
     46     lct[fa].fa=x;son(x,!t)=fa;
     47     lct[x].fa=fafa;
     48     if(!lct[fa].is_root)son(fafa,son(fafa,1)==fa)=x;
     49     else lct[x].is_root=1,lct[fa].is_root=0;
     50     push_up(fa);
     51     push_up(x);
     52   }
     53   void splay(int x){
     54     push(x);
     55     for(int fa;!lct[x].is_root;rotate(x))
     56       if(!lct[fa=lct[x].fa].is_root)
     57     rotate(getson(fa)==getson(x)?fa:x);
     58   }
     59   void access(int x){
     60     int y=0;
     61     do{
     62       splay(x);
     63       lct[rr(x)].is_root=1;
     64       lct[rr(x)=y].is_root=0;
     65       push_up(x);
     66       x=lct[y=x].fa;
     67     }while(x);
     68   }
     69   void mroot(int x){
     70     access(x);
     71     splay(x);
     72     push_rev(x);
     73   }
     74   void link(int u,int v){
     75     mroot(u);
     76     lct[u].fa=v;
     77   }
     78   void cut(int u,int v){
     79     mroot(u);
     80     access(v);splay(v);
     81     lct[ll(v)].fa=lct[v].fa;
     82     lct[ll(v)].is_root=1;
     83     ll(v)=lct[v].fa=0;
     84     push_up(v);
     85   }
     86   void change(int x,int y){
     87     mroot(x);
     88     lct[x].x=y;
     89     push_up(x);
     90   }//修改点
     91   bool judge(int x,int y){
     92     access(x);splay(x);while(ll(x))x=ll(x);
     93     access(y);splay(y);while(ll(y))y=ll(y);
     94     return x==y;
     95   }
     96   bool pd(int x,int y){
     97     mroot(x);
     98     access(y);splay(y);
     99     return (ll(y)==x)&&lct[ll(y)].size==1;
    100   }
    101   int query(int x,int y){
    102     mroot(x);
    103     access(y);splay(y);
    104     return lct[y].mmax;
    105   }
    106 }T[10];
    107 struct node{
    108   int next,to,dis;
    109 }edge[200005];
    110 int head[10005],size=-1;
    111 void putin(int from,int to,int dis){
    112   size++;
    113   edge[size].next=head[from];
    114   edge[size].to=to;
    115   edge[size].dis=dis;
    116   head[from]=size;
    117 }
    118 int main(){
    119   int i,j;
    120   memset(head,-1,sizeof(head));
    121   scanf("%d%d%d%d",&n,&m,&C,&k);
    122   for(i=1;i<=n;i++){
    123     int x;scanf("%d",&x);
    124     for(j=0;j<C;j++){
    125       T[j].lct[i].size=T[j].lct[i].is_root=1;
    126       T[j].lct[i].x=T[j].lct[i].mmax=x;
    127     }
    128   }
    129   for(i=1;i<=m;i++){
    130     int u,v,c;
    131     scanf("%d%d%d",&u,&v,&c);
    132     putin(u,v,c);
    133     putin(v,u,c);
    134     in[u][c]++;
    135     in[v][c]++;
    136     T[c].link(u,v);
    137   }
    138   for(i=1;i<=k;i++){
    139     int a,b,c,d;
    140     scanf("%d%d%d",&d,&a,&b);
    141     if(d==0){
    142       
    143       for(j=0;j<C;j++)
    144     T[j].change(a,b);
    145       
    146     }
    147     else if(d==1){
    148       
    149       scanf("%d",&c);
    150       
    151       bool flag=0;
    152       for(j=head[a];j!=-1;j=edge[j].next)if(edge[j].to==b){flag=1;break;}
    153       if(!flag){printf("No such edge.
    ");continue;}
    154       
    155       int last=edge[j].dis;
    156       if((in[a][c]==2&&last!=c)||(in[b][c]==2&&last!=c)){printf("Error 1.
    ");continue;}
    157 
    158       if(T[c].judge(a,b)&&(!T[c].pd(a,b))){printf("Error 2.
    ");continue;}
    159       
    160       in[a][last]--;in[b][last]--;
    161       in[a][c]++;in[b][c]++;
    162       edge[j].dis=c;edge[j^1].dis=c;
    163       T[last].cut(a,b);
    164       T[c].link(a,b);
    165       printf("Success.
    ");
    166       
    167     }
    168     else if(d==2){
    169       
    170       scanf("%d",&c);
    171       
    172       if(!T[a].judge(b,c))printf("-1
    ");
    173       else printf("%d
    ",T[a].query(b,c));
    174       
    175     }
    176   }
    177   return 0;
    178 }

     

  • 相关阅读:
    selenium 学习(-)
    阿里百川SDK初始化失败 错误码是203
    sql-如何提高SQL查询的效率?
    索引失效原因总结
    索引失效原因总结
    sql-优化-大表关联小表
    sql-优化-大表关联小表
    C#-How to get series point info when mouse is over chartcontrol
    C#-How to get series point info when mouse is over chartcontrol
    C#-ILookup 初始化
  • 原文地址:https://www.cnblogs.com/huangdalaofighting/p/8301213.html
Copyright © 2011-2022 走看看