无根树同构_hash
先贴上地址 https://vjudge.net/problem/HDU-5732
判断有根树同构:
1. 直接用括号最小表示法
2. 利用括号最小表示法的思想进行hash
判断无根树同构:
1. 找到树的重心.
2. 以重心为根, 把无根树转化成有根树. 按照有根树同构的方法判断是否同构.
同构的过程中,为什么可以sort.
我们知道,对于树来说,
树的节点绕着它的父节点旋转,树的结构就不会被改变的.
所以sort的过程就相当于把树的节点绕着它的父节点进行旋转.
// sort的话,可以这么理解:
我们是按照同样的规则(我们保证这个规则可以唯一确定一棵树,),
对两棵树进行操作,如果同构的话,那么结果应该是一样的. 如果不同构的画 那么结果就是不一样的
至于树的括号表达式, 可以见这几个博客
括号表达式 https://www.byvoid.com/zhs/blog/directed-tree-bracket-sequence
树同构 https://blog.csdn.net/u010152669/article/details/9116975
树的表示方法 https://www.cnblogs.com/jsawz/p/6807636.html
#include <cstdio>
#include <ctime>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <map>
#include <vector> using namespace std; typedef unsigned long long ull;
const int maxn = 1e5+;
ull x[maxn];
int ans[maxn]; struct Edge {
int lst;
int to;
}; class Tree {
public :
Edge edge[maxn<<];
int head[maxn];
int cn, csz, ccnt, mid[];
int rcd[maxn];
ull cnode[maxn];
ull vvvvv[];
map<string, int> id;
char name[maxn][]; inline void add(int u, int v) {
edge[csz].lst = head[u];
edge[csz].to = v;
head[u] = csz++;
} int dfs(int u, int fa) {
int i, v, res = , t1;
for (i=head[u]; i; i=edge[i].lst) {
v = edge[i].to;
if (v == fa) continue;
t1 = dfs(v, u);
res += t1;
if (rcd[u] < t1) rcd[u] = t1;
if (rcd[v] < cn-t1) rcd[v] = cn-t1;
}
return res + ;
} ull dfs2(int u, int fa, int deep) {
int i, v, res = , t1;
vector<ull> son;
for (i=head[u]; i; i=edge[i].lst) {
v = edge[i].to;
if (v == fa) continue;
son.push_back(dfs2(v, u, deep+));
}
sort(son.begin(), son.end()); // 判断树同构 注意要sort后乘一个随机数(或者一个质数的i次方). 为什么可以sort呢? 因为树同一层的节点围绕父节点旋转不会改变树的结构. 而sort相当于把节点绕父节点旋转了.
for (i=, t1=son.size(); i<t1; ++i)
res += son[i] * x[i+];
return cnode[u] = (res ? res : x[deep]);
} void init(int n) {
cn = n; csz = ; id.clear();
int i, u, v, cnt = ;
char s1[], s2[]; for (i=; i<=n; ++i) rcd[i] = head[i] = ; for (i=; i<n; ++i) {
scanf("%s%s", s1, s2);
if (!(u = id[s1])) {
strcpy(name[cnt], s1);
id[s1] = u = cnt++;
} if (!(v = id[s2])) {
strcpy(name[cnt], s2);
id[s2] = v = cnt++;
}
add(u, v); add(v, u);
} dfs(, -);
int mm = 0x3f3f3f3f;
for (i=; i<=n; ++i) {
if (mm > rcd[i]) {
mm = rcd[i];
ccnt = ;
mid[++ccnt] = i;
} else if (mm == rcd[i]) mid[++ccnt] = i;
} for (i=; i<=ccnt; ++i)
vvvvv[i] = dfs2(mid[i], -, );
}
}te1, te2; struct nobe {
int id;
ull val;
bool operator < (const nobe &a) const {
return val < a.val;
}
nobe () {}
nobe (int iid, ull vval) : id(iid), val(vval) {}
}; void dfs3(int u1, int u2, int fa1, int fa2) {
int i, v1, v2, tsz;
ans[u1] = u2;
vector<nobe> ve1, ve2;
for (i=te1.head[u1]; i; i=te1.edge[i].lst) {
v1 = te1.edge[i].to;
if (v1 == fa1) continue;
ve1.push_back(nobe(v1, te1.cnode[v1]));
}
for (i=te2.head[u2]; i; i=te2.edge[i].lst) {
v2 = te2.edge[i].to;
if (v2 == fa2) continue;
ve2.push_back(nobe(v2, te2.cnode[v2]));
}
sort(ve1.begin(), ve1.end());
sort(ve2.begin(), ve2.end());
for (i=, tsz=ve1.size(); i<tsz; ++i)
dfs3(ve1[i].id, ve2[i].id, u1, u2);
} int main()
{
int i, j, k, n;
srand(time(NULL));
for (i=; i<maxn; ++i) x[i] = rand();
while (~scanf("%d", &n)) {
te1.init(n);
te2.init(n);
for (i=; i<=te1.ccnt; ++i)
for (j=; j<=te2.ccnt; ++j) {
if (te1.vvvvv[i] == te2.vvvvv[j]) {
dfs3(te1.mid[i], te2.mid[j], te1.mid[i], te2.mid[j]);
for (k=; k<=n; ++k)
printf("%s %s\n", te1.name[k], te2.name[ans[k]]);
goto A;
}
}
A: ;
} return ;
}
无根树同构_hash的更多相关文章
- 3194. 【HNOI模拟题】化学(无标号无根树计数)
Problem 求\(n\)个点的每个点度数不超过\(4\)的无标号无根树个数. Data constraint \(1\le n\le 500\) Solution 尝试着把问题一般化.我们来考虑一 ...
- BZOJ 4754 [JSOI2016]独特的树叶 | 树哈希判同构
题目链接 这道题是一道判断无根树同构的模板题,判断同构主要的思路就是哈希. 一遇到哈希题,一百个人能有一百零一种哈希方式,这篇题解随便选用了一种--类似杨弋<Hash在信息学竞赛中的一类应用&g ...
- 【CF1252F】Regular Forestation(重心,树同构)
题意:给定一棵n个点的树,问删去某个点之后所有的树同构,这样分割出来的树最多能有几棵 n<=4000 思路:分割成至少两个size相等的联通块之后size必定小于n/2,与树的重心的定义相同 预 ...
- 【BZOJ4337】树的同构(树同构,哈希)
题意: 树是一种很常见的数据结构. 我们把N个点,N-1条边的连通无向图称为树. 若将某个点作为根,从根开始遍历,则其它的点都有一个前驱,这个树就成为有根树. 对于两个树T1和T2,如果能够把树T1T ...
- ustc 1117
无根树同构 有两种方法,一种是确定其中一棵树,另一棵树枚举根节点. 一种是,利用拓扑排序,先确定其中一棵树.另一棵树,若拓扑后剩两个节点,则枚举这两个节点为根结点,否则,只需做一次.注意,无根树节点入 ...
- 树hash
判断树的同构,采用树hash的方式. 树hash定义在有根树上.判断无根树同构的时候,可以比较重心为根的hash值或者比较每个点为根的hash值. h[x]表示x为根的子树的hash,g[x]表示x为 ...
- uva12489 Combating cancer(树同构)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud https://uva.onlinejudge.org/index.php?opt ...
- 4337: BJOI2015 树的同构
题解: 树的同构的判定 有根树从根开始进行树hash 先把儿子的f进行排序 $f[i]=\sum_{j=1}^{k} { f[j]*prime[j]} +num[i]$(我没有仔细想这样是不是树是唯一 ...
- [BJOI2015]树的同构
嘟嘟嘟 判断树的同构的方法就是树上哈希. 如果树是一棵有根树,那么只要从根节点出发dfs,每一个节点的哈希值等于按传统方式算出来的子树的哈希值的结果.需要注意的是,算完子树的哈希值后要先排序再加起来, ...
随机推荐
- FJUT16级第一周寒假作业题解I题
涨姿势题3 TimeLimit:1000ms MemoryLimit:128000KB 64-bit integer IO format:%lld Problem Description 涨姿势题就 ...
- qt 调用zlib压缩与解压缩功能
Zlib是一种免费且通用的压缩库,由于Zlib压缩效果比LZW好,而且解压缩速度快,更重要的是商业软件中使用Zlib不需要缴纳版权费,所以很多游戏都使用Zlib压缩资源文件. Zlib是由Jean-l ...
- Python3各种进制之间的转换方法
一.2/8/10/16进制互转 1.1 2/8/10/16进制赋值 # 二进制赋值以0b打头 a = 0b1000 # 八进制赋值以0o打头,第一个是数字0第二个是字母o b = 0o1100 # 十 ...
- anglar cli的 rxjs_1.of is not a function
按照官网敲例子遇到 rxjs_1.of is not a function import { Observable,of } from 'rxjs'; 改为 import { Observable} ...
- CAD(布置厨洁具)(尺寸标注)5.12
"TYTK"打开图库,找到平面厨具和洁具.双击选中的厨具,A可以不停旋转90度.给厨具选取正确的位置.画出灶台线,同理画出卫生间的家具.绘制出洗脸台的平台.浴缸的平台. 尺寸标注: ...
- bzoj3930
题解: 莫比乌斯函数 然而向我这种弱菜肯定选择暴力dp 代码: #include<bits/stdc++.h> ,M=; typedef long long ll; using names ...
- day041 前端HTML CSS基本选择器(未整理完毕)
标签: <b> :加粗 <i> :倾斜体 <u>: 下划线 <s>: 删除线 <p>:段落 <h1> - <h6> ...
- MyBatis逆向工程:根据table生成Model、Mapper、Mapper.xml
逆向工程工具 下载地址:https://download.csdn.net/download/zhutouaizhuwxd/10779140 1.工程导入Eclipse 2.运行MainUI.jav ...
- 二:通过VirtualBox+Vagrant创建一个centos的虚拟机:
官网安装VirtualBox及Vagrant. 下载centos7,添加到vagrant中. http://e-proxy.yfb.sunline.cn/download/vagrant/centos ...
- 关于Xcode9 无法读取文件的问题
以前我们加载本地文件的时候也许没有注意,可是在Xcode9中会出现许多问题,经常会出现图片无法显示,本地html无法加载等问题: 当然不是Xcode的问题,只是以前我们并没有注意,其实Xcode对这些 ...