题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4298

题面:

  给定d张无向图,每张图都有n个点。一开始,在任何一张图中都没有任何边。接下来有m次操作,每次操作会给出a,b,k,意为在第k张图中的点a和点b之间添加一条无向边。你需要在每次操作之后输出有序数对(a,b)的个数,使得1<=a,b<=n,且a点和b点在d张图中都连通。

  第一行包含三个正整数d,n,m(1<=d<=200,1<=n<=5000,1<=m<=1000000),依次表示图的个数,点的个数和操作的个数。
  接下来m行,每行包含三个正整数a,b,k(1<=a,b<=n,1<=k<=d),依次描述每一个操作。
  输出m行m个正整数,依次表示每次操作之后满足条件的有序数对(a,b)的个数。

与连通性有关,那么就是并查集咯。

但在一个图里连通了两个点集之后,难道要遍历该点集所有点对,看看在其他图里是否连通?

1.启发式合并。那么对答案的影响,可以考虑落在 “新加入该点集” 的那些点上。

2.考虑 “与一个点在 d 张图里都连通的点的个数” 。

  连通意味着处在同一个连通块中。那么记录每个点在 d 张图中分别属于哪些连通块,这形成一个字符串。

  哈希记录某个字符串对应的点数即可。

需要用哈希表,而且消失的哈希值要从哈希表里删除,才能不超时。即使这样,自己还是平均时间的两倍。

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
#define ll long long
#define ull unsigned long long
#define pb push_back
using namespace std;
int rdn()
{
int ret=;bool fx=;char ch=getchar();
while(ch>''||ch<''){if(ch=='-')fx=;ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return fx?ret:-ret;
}
int g[];
void wrt(ll x)
{
if(!x){puts("");return;}
int t=;
while(x)g[++t]=x%,x/=;
for(int i=t;i;i--)putchar(g[i]+'');puts("");
}
const int N=,M=,bs=,M2=N*M*;
int n,m,D,fa[N][M],siz[N][M],ct[M2];
ll ans; ull bin[M],hs[N];
vector<int> vt[N][M];
namespace H{
const int md=3e7;
int hd[md],xnt,nxt[M2]; ull to[M2];
int get(ull x)
{
int h=x%md;
for(int i=hd[h];i;i=nxt[i])
if(to[i]==x)return i;
to[++xnt]=x; nxt[xnt]=hd[h]; hd[h]=xnt;
return xnt;
}
void del(ull x)
{
int h=x%md;
if(to[hd[h]]==x){hd[h]=nxt[hd[h]];return;}
for(int i=hd[h],lst=;i;lst=i,i=nxt[i])
if(to[i]==x){nxt[lst]=nxt[i];break;}
}
}
int main()
{
D=rdn();n=rdn();m=rdn();
bin[]=;for(int i=;i<=D;i++)bin[i]=bin[i-]*bs;
for(int i=;i<=n;i++)
{
for(int j=;j<=D;j++)
{
fa[i][j]=i; siz[i][j]=;
vt[i][j].pb(i); hs[i]+=i*bin[j];
}
ct[H::get(hs[i])]++;
}
ans=n;
for(int i=,u,v,d;i<=m;i++)
{
u=rdn();v=rdn();d=rdn();
u=fa[u][d]; v=fa[v][d];
if(u==v){wrt(ans);continue;}
if(siz[u][d]<siz[v][d])swap(u,v);
int tot=siz[u][d];
siz[u][d]+=siz[v][d]; vt[u][d].resize(siz[u][d]);
ull tmp=(u-v)*bin[d];
for(int j=,cr;j<siz[v][d];j++)
{
cr=vt[v][d][j]; fa[cr][d]=u; vt[u][d][tot++]=cr;
ull x=hs[cr],y=hs[cr]+tmp; hs[cr]+=tmp;
int t0=H::get(x),t1=H::get(y);
ct[t0]--; ans-=ct[t0]*; ans+=ct[t1]*; ct[t1]++;
}
vector<int> ().swap(vt[v][d]);
wrt(ans);
}
return ;
}

bzoj 4298 [ONTAK2015]Bajtocja——哈希+启发式合并的更多相关文章

  1. BZOJ.4298.[ONTAK2015]Bajtocja(Hash 启发式合并)

    题目链接 \(Description\) 给定\(d\)张无向图,每张图都有\(n\)个点.一开始,在任何一张图中都没有任何边. 接下来有\(m\)次操作,每次操作会给出\(a,b,k\),意为在第\ ...

  2. @bzoj - 4298@ [ONTAK2015]Bajtocja

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定d张无向图,每张图都有n个点.一开始,在任何一张图中都没有任 ...

  3. BZOJ 2733: [HNOI2012]永无乡 启发式合并treap

    2733: [HNOI2012]永无乡 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  4. BZOJ 1483: [HNOI2009]梦幻布丁( 链表 + 启发式合并 )

    把相同颜色的串成一个链表, 然后每次A操作就启发式合并, 然后计算对答案的影响. ----------------------------------------------------------- ...

  5. BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]

    3123: [Sdoi2013]森林 题意:一个森林,加边,询问路径上k小值.保证任意时刻是森林 LCT没法搞,树上kth肯定要用树上主席树 加边?启发式合并就好了,小的树dfs重建一下 注意 测试点 ...

  6. bzoj 1483: [HNOI2009]梦幻布丁 (链表启发式合并)

    Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色. 例如颜色分别为1,2,2,1的四个布丁一共有3段颜色. Input ...

  7. BZOJ 2733 [HNOI2012]永无乡 - 启发式合并主席树

    Description 1: 查询一个集合内的K大值 2: 合并两个集合 Solution 启发式合并主席树板子 Code #include<cstdio> #include<cst ...

  8. bzoj 2809 左偏树\平衡树启发式合并

    首先我们对于一颗树,要选取最多的节点使得代价和不超过m,那么我们可以对于每一个节点维护一个平衡树,平衡树维护代价以及代价的和,那么我们可以在logn的时间内求出这个子树最多选取的节点数,然后对于一个节 ...

  9. BZOJ 4919: [Lydsy1706月赛]大根堆 启发式合并

    我不会告诉你这是线段树合并的好题的... 好吧我们可以搞一个multiset在dfs时求出LIS(自带二分+排序)进行启发式合并,轻松加愉悦... #include<cstdio> #in ...

随机推荐

  1. 成功秀了一波scala spark ML逻辑斯蒂回归

    1.直接上官方代码,调整过的,方可使用 package com.test import org.apache.spark.{SparkConf, SparkContext} import org.ap ...

  2. 文档压缩 | gzip、bzip2、xz

    6.文档的压缩与打包 Linux下常见后缀名所对应的的压缩工具 .gz 表示由gzip压缩工具压缩的文件 .bz2 表示由bzip2压缩工具压缩的文件 .tar 表示由tar打包程序打包的文件(tar ...

  3. linux ftp使用相关

    ftp 7.7.6.201 21121 name:aaa password:123456

  4. SQL Server中的扩展事件学习系列

    SQL Server 扩展事件(Extented Events)从入门到进阶(1)——从SQL Trace到Extented Events SQL Server 扩展事件(Extented Event ...

  5. 数塔 Medium

    Summer is coming! It's time for Iahub and Iahubina to work out, as they both want to look hot at the ...

  6. HNUSTOJ 1601:名字缩写

    1601: 名字缩写 时间限制: 1 Sec  内存限制: 128 MB 提交: 288  解决: 80 [提交][状态][讨论版] 题目描述 Noname老师有一个班的学生名字要写,但是他太懒了,想 ...

  7. Ajax请求后台数据

    一.前期准备 安装好XAMPP软件,并运行起来.本文代码是基于XAMPP开发环境,XAMPP是完全免费且易于安装的Apache发行版,其中包含MariaDB.PHP和Perl.XAMPP开放源码包的设 ...

  8. 原生js事件委托(事件代理)方法扩展

    原生js事件委托(事件代理)方法扩展: 通过Node底层原型扩展委托方法 /** * 事件委托方法 * @param eventName {string}:事件名称,如'click' * @param ...

  9. PCIe事务层の详解(一)

    PCIe总线的通信机制:当一个设备要想另一个设备进行读取通信时,请求方requester需要向另一个设备发送请求request,靶向方作为事件完成方completer,以complete Packet ...

  10. 完整ASP.Net Excel导入

    //把EXCEL文件上传到服务器并返回文件路径        private String typename(FileUpload fileloads)        {            str ...