无根树同构_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,每一个节点的哈希值等于按传统方式算出来的子树的哈希值的结果.需要注意的是,算完子树的哈希值后要先排序再加起来, ...
随机推荐
- Swagger 路径过滤 -PreSerializeFilters
Swagger 默认显示所有api, 如果要做路径过滤,可以这样做. //过滤,只显示部分api app.UseSwagger(c=> { c.PreSerializeFilters.Add(( ...
- Java 9中新的货币API
译文出处: Java译站 原文出处:Michael Scharhag JSR 354定义了一套新的Java货币API,计划会在Java 9中正式引入.本文中我们将来看一下它的参考实现:JavaMo ...
- 巧用call,appl有 根据对象某一属性求最大值
查找对象数组中某属性的最大最小值的快捷方法 例如要查找array数组中对象的value属性的最大值 var array=[ { "index_id": 119, "are ...
- 各种形式的熵函数,KL距离
自信息量I(x)=-log(p(x)),其他依次类推. 离散变量x的熵H(x)=E(I(x))=-$\sum\limits_{x}{p(x)lnp(x)}$ 连续变量x的微分熵H(x)=E(I(x)) ...
- laravel中的storePublicly对上传的文件设置上传途径
public function imgeUpload(Request $request) { //生成的文件名是md5随机的文件名字 //$path=$request->file('wangEd ...
- mybatis实战教程
参考:http://blog.csdn.net/techbirds_bao/article/details/9233599/
- Django ajax提交 登录
一.url from django.contrib import adminfrom django.urls import pathfrom appo1 import views urlpattern ...
- jenkins+findbugs+checkstyle+PMD静态代码检查(二)
可以根据自己的需求选中对应的插件进行配置(不一定非要同时配置三个插件) jenkins:持续集成的工具 fundbugs:检测代码静态错误的插件 例如:定义了没有用到的对象,string类型的比较使 ...
- mysql 数据迁移
最近线上系统新挂了一次磁盘,需要将系统磁盘下的 mysql 数据目录迁移到 数据盘上. 经过一番考察,mysql在安装时,使用了预编译的二进制tar.gz包.共有两处配置了 datadir属性 /et ...
- SignalR NuGet程序包
最近公司有一个边看直播边聊天的需求,直播好搞,直接用腾讯的小直播,组装推流和播放地址,把推流地址拿出去就OK,只要一推流,就可以使用播放地址观看直播,看完后通过webclient去异步下载直播的视频到 ...