zoukankan      html  css  js  c++  java
  • Graph HDU

    https://vjudge.net/problem/HDU-4467

    大概就是,设一个块大小T

    对于度数<=T的点,设为1类点,在改变颜色的时候暴力查询与其相邻点,更新答案

    对于度数>T的点,设为2类点,分别维护与其相邻的颜色为0/1的点之间的边权和(记录与每个点相连的所有2类点,然后在任意点(注意2类点也要)改变颜色时维护一下与其相连2类点的这个值),改变颜色的时候根据维护的值O(1)可以计算出对答案的修改

    说的再简单一点,1类点由自身去更新其他,2类点由其他去更新自身。。复杂度很容易发现是根号级别的(T=sqrt(m))

    但是要注意,重边一定要去。。。不然可以被卡到$n^2$

    错误记录:

    1.没有在改变2类点颜色时改变相连其他2类点维护的值

    2.没有去重边

    3.d[a],d[b]和a,b没有极其仔细地区分

      1 //#pragma GCC optimize(3)
      2 #include<cstdio>
      3 #include<algorithm>
      4 #include<cstring>
      5 #include<vector>
      6 #include<set>
      7 #include<map>
      8 using namespace std;
      9 #define fi first
     10 #define se second
     11 #define mp make_pair
     12 #define pb push_back
     13 typedef long long ll;
     14 typedef unsigned long long ull;
     15 typedef pair<int,int> pii;
     16 #define N 100000
     17 struct E
     18 {
     19     int to,nxt;ll d;
     20 }e[2*N+100];
     21 int f1[N+100],ne;
     22 int TT,n,m,mm,d[N+100];
     23 int in[N+100],sz,q;
     24 //1类点:in[i]<=sz;2类点:in[i]>sz
     25 ll a1[N+100][2];
     26 //a1[i][j]:i为2类点,i与相邻的颜色为j的点之间的所有边的权值和
     27 //int num[100100];
     28 char tmp[10];
     29 vector<int> ss[N+100];//ss[i]:i点到相邻的2类点的边
     30 ll ans[3];//ans[0]:00边答案;ans[1]:11边答案;ans[2]:01边答案
     31 map<pii,pii> ma;
     32 ll &gg(int a,int b)
     33 {
     34     if(a==0&&b==0)    return ans[0];
     35     if(a==1&&b==1)    return ans[1];
     36     return ans[2];
     37 }
     38 int main()
     39 {
     40     int a,b,c,i,k;pii t;
     41     while(scanf("%d%d",&n,&m)==2)
     42     {
     43         TT++;mm=0;//num[0]=0;
     44         memset(f1,0,sizeof(f1));ne=0;ma.clear();
     45         memset(in,0,sizeof(in));ans[0]=ans[1]=ans[2]=0;
     46         memset(a1,0,sizeof(a1));
     47         for(i=1;i<=n;i++)    scanf("%d",&d[i]);
     48         for(i=1;i<=m;i++)
     49         {
     50             scanf("%d%d%d",&a,&b,&c);
     51             if(a>b)    swap(a,b);
     52             if(ma.count(mp(a,b)))
     53             {
     54                 t=ma[mp(a,b)];
     55                 e[t.fi].d+=c;e[t.se].d+=c;
     56             }
     57             else
     58             {
     59                 e[++ne].to=b;e[ne].nxt=f1[a];f1[a]=ne;e[ne].d=c;
     60                 e[++ne].to=a;e[ne].nxt=f1[b];f1[b]=ne;e[ne].d=c;
     61                 in[a]++;in[b]++;
     62                 ma[mp(a,b)]=mp(ne-1,ne);mm++;
     63             }
     64             gg(d[a],d[b])+=c;
     65         }
     66         sz=sqrt(2*mm);
     67         for(i=1;i<=n;i++)
     68             if(in[i]>sz)
     69             {
     70                 //num[++num[0]]=i;
     71                 for(k=f1[i];k;k=e[k].nxt)
     72                     a1[i][d[e[k].to]]+=e[k].d;
     73             }
     74         for(i=1;i<=n;i++)
     75         {
     76             ss[i].clear();
     77             for(k=f1[i];k;k=e[k].nxt)
     78                 if(in[e[k].to]>sz)
     79                     ss[i].pb(k);
     80         }
     81         scanf("%d",&q);
     82         printf("Case %d:
    ",TT);
     83         while(q--)
     84         {
     85             scanf("%s",tmp);
     86             if(tmp[0]=='A')
     87             {
     88                 scanf("%d%d",&a,&b);
     89                 printf("%lld
    ",gg(a,b));
     90             }
     91             else
     92             {
     93                 scanf("%d",&a);
     94                 if(in[a]<=sz)
     95                 {
     96                     for(k=f1[a];k;k=e[k].nxt)
     97                     {
     98                         b=e[k].to;
     99                         gg(d[a],d[b])-=e[k].d;
    100                         if(in[b]>sz)    a1[b][d[a]]-=e[k].d;
    101                     }
    102                     d[a]^=1;
    103                     for(k=f1[a];k;k=e[k].nxt)
    104                     {
    105                         b=e[k].to;
    106                         gg(d[a],d[b])+=e[k].d;
    107                         if(in[b]>sz)    a1[b][d[a]]+=e[k].d;
    108                     }
    109                 }
    110                 else
    111                 {
    112                     for(i=0;i<ss[a].size();i++)
    113                         a1[e[ss[a][i]].to][d[a]]-=e[ss[a][i]].d;
    114                     gg(d[a],0)-=a1[a][0];gg(d[a],1)-=a1[a][1];
    115                     d[a]^=1;
    116                     gg(d[a],0)+=a1[a][0];gg(d[a],1)+=a1[a][1];
    117                     for(i=0;i<ss[a].size();i++)
    118                         a1[e[ss[a][i]].to][d[a]]+=e[ss[a][i]].d;
    119                 }
    120             }
    121         }
    122     }
    123     return 0;
    124 }
  • 相关阅读:
    actionbar中添加searchview并监听期伸缩/打开的方法
    Java基础学习总结(87)——坚持写Java等技术类博客的好处
    Nginx学习总结(3)——Nginx配置及应用场景之高级配置
    [置顶] 软件版本命名规范及各阶段说明
    戏说云计算之PaaS,IaaS,SaaS
    App后台开发运维和架构实践学习总结(4)——APP的注册和登录功能设计
    Maven学习总结(30)——Maven项目通用三级版本号说明
    Java Web学习总结(29)——Java Web中的Filter和Interceptor比较
    养成10个优秀的习惯
    Java基础学习总结(86)——Java异常处理机制Exception抛出异常时throw和throws用法详解
  • 原文地址:https://www.cnblogs.com/hehe54321/p/9327153.html
Copyright © 2011-2022 走看看