先贴上地址 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的更多相关文章

  1. 3194. 【HNOI模拟题】化学(无标号无根树计数)

    Problem 求\(n\)个点的每个点度数不超过\(4\)的无标号无根树个数. Data constraint \(1\le n\le 500\) Solution 尝试着把问题一般化.我们来考虑一 ...

  2. BZOJ 4754 [JSOI2016]独特的树叶 | 树哈希判同构

    题目链接 这道题是一道判断无根树同构的模板题,判断同构主要的思路就是哈希. 一遇到哈希题,一百个人能有一百零一种哈希方式,这篇题解随便选用了一种--类似杨弋<Hash在信息学竞赛中的一类应用&g ...

  3. 【CF1252F】Regular Forestation(重心,树同构)

    题意:给定一棵n个点的树,问删去某个点之后所有的树同构,这样分割出来的树最多能有几棵 n<=4000 思路:分割成至少两个size相等的联通块之后size必定小于n/2,与树的重心的定义相同 预 ...

  4. 【BZOJ4337】树的同构(树同构,哈希)

    题意: 树是一种很常见的数据结构. 我们把N个点,N-1条边的连通无向图称为树. 若将某个点作为根,从根开始遍历,则其它的点都有一个前驱,这个树就成为有根树. 对于两个树T1和T2,如果能够把树T1T ...

  5. ustc 1117

    无根树同构 有两种方法,一种是确定其中一棵树,另一棵树枚举根节点. 一种是,利用拓扑排序,先确定其中一棵树.另一棵树,若拓扑后剩两个节点,则枚举这两个节点为根结点,否则,只需做一次.注意,无根树节点入 ...

  6. 树hash

    判断树的同构,采用树hash的方式. 树hash定义在有根树上.判断无根树同构的时候,可以比较重心为根的hash值或者比较每个点为根的hash值. h[x]表示x为根的子树的hash,g[x]表示x为 ...

  7. uva12489 Combating cancer(树同构)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud https://uva.onlinejudge.org/index.php?opt ...

  8. 4337: BJOI2015 树的同构

    题解: 树的同构的判定 有根树从根开始进行树hash 先把儿子的f进行排序 $f[i]=\sum_{j=1}^{k} { f[j]*prime[j]} +num[i]$(我没有仔细想这样是不是树是唯一 ...

  9. [BJOI2015]树的同构

    嘟嘟嘟 判断树的同构的方法就是树上哈希. 如果树是一棵有根树,那么只要从根节点出发dfs,每一个节点的哈希值等于按传统方式算出来的子树的哈希值的结果.需要注意的是,算完子树的哈希值后要先排序再加起来, ...

随机推荐

  1. PowerShell使用教程

    一.说明 1.1 背景说明 个人对PowerShell也不是很熟悉,开始的时候就突然看到开始菜单中多了个叫PowerShell的文件夹,后来一点就看到某个教程视频说PowerShell很厉害但也没怎么 ...

  2. nop 4.1 Widget 探究- 视图组件

    1. 系统默认自带了一个NivoSlider 的Wdget. 在Nop.Web项目首页的HomePageTop里 这个写法是 ASP.NET Core MVC 中的新特性 视图组件,与局部视图相似,但 ...

  3. 判断window.open的页面是否已经被关

    <!DOCTYPE html><html><head><meta charset="utf-8"><title>菜鸟教程 ...

  4. 解决eclipse/myeclipse导入项目时出现红色叹号的方法

  5. shell 通配符

    Bash中的通配符 '?' 匹配一个任意字符 '*' 匹配0个或任意多个字符,也就是可以匹配任何内容 '[]' 匹配括号中任意一个字符.例如[abc]代表一定匹配一个字符,或者是a,或者是b,或者是c ...

  6. Java中的equals和==的差别 以及Java中等价性和同一性的讨论

    ==对基本数据类型比较的是值,对引用类型比较的是地址 equals()比较的是对象的数据的引用 等价性原理: 自反性    x.equals(x)为true 对称性    x.equals(y) 为t ...

  7. Tomcat错误:getOutputStream() has already been called for this response

    使用weblogic部署时,没有报错.客户现场使用tomcat后报错. 在tomcat下jsp中出现此错误一般都是在jsp中使用了输出流(如输出图片验证码,文件下载等),没有妥善处理好的原因.具体的原 ...

  8. bzoj2045

    题解: 莫比乌斯反演经典题目 直接套公式好了 代码: #include<bits/stdc++.h> using namespace std; ; typedef long long ll ...

  9. Oracle 11gR2 Database UNDO表空间使用率居高不下处理

    一.UNDO表空间监控图 Prometheus监控的到UNDO表空间使用率超过90%(90%为所有表空间告警阈值).从图中可以看到,多次增加UNDO表空间的DATAFILE,UNDO表空间达到40GB ...

  10. C++基础知识:操作符重载

    1.C++标准库: C++标准库并不是C++语言的一部分C++标准库是由C++语言编写而成的类库和函数的集合C++标准库中定义的类和对象都位于std命名空间中C++标准库的头文件都不带.h后缀C++标 ...