[海军国际项目办公室]数树

2024-03-16 22:38

本文主要是介绍[海军国际项目办公室]数树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

数树

题目概述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

题解

我们先不考虑树 T 2 T2 T2树内部的同构情况,算出总的方案数,最后除去同构的方案。
显然,树 T 1 T1 T1 T 2 T2 T2都是无根树,我们不妨先给 T 1 T1 T1定一个根,这样的话, T 1 T1 T1中每一种能映射到 T 2 T2 T2上的合法方案都会存在一个确定的根,在我们将我们把 T 2 T2 T2的根定义为该对应点时,这两块刚好能够匹配。
我们不妨枚举在 T 2 T2 T2中每个点为根时的情况,这样所有点的相对高度与位置都是确定的,我们可以把它放到 T 1 T1 T1上,看有多少个连通块能跟其匹配。

T 2 T2 T2放到 T 1 T1 T1求出匹配显然可以用 d p dp dp来求出。
我们定义 d p i , j dp_{i,j} dpi,j表示 T 1 T1 T1上的点 i i i匹配 T 2 T2 T2上的点 j j j的方案数。
其转移过程我们可以通过状压来维护。
由于我们此时 T 2 T2 T2的根是固定的,所以点 j j j T 2 T2 T2上的儿子也是固定的。
我们要让 i i i T 1 T1 T1上的儿子依次匹配 j j j T 2 T2 T2上的儿子,求出匹配的方案数。
我们定义 f i , j , S f_{i,j,S} fi,j,S表示 i i i T 1 T1 T1上的儿子匹配了 j j j T 2 T2 T2上的儿子集合为 S S S的方案数,其转移显然是一个背包的过程,加入的 i i i的儿子,枚举其匹配 j j j的哪一个儿子或者不匹配。
显然, d p i , j = f i , j , s o n j dp_{i,j}=f_{i,j,son_{j}} dpi,j=fi,j,sonj,其中 s o n j son_{j} sonj表示 j j j儿子的全集。
∑ i = 1 n d p i , r t \sum_{i=1}^{n}dp_{i,rt} i=1ndpi,rt就是 T 2 T2 T2 r t rt rt为根放到 T 1 T1 T1上匹配得到的方案数之和。
我们要枚举 r t rt rt,整个 d p dp dp的过程是 O ( n m 2 2 m ) O\left(nm^22^m\right) O(nm22m)的。

但是 T 2 T2 T2的内部存在同构的情况,比如一棵完全二叉树,交换它的左右儿子,树的形态并没有发生改变,但如果是我们上面的匹配过程的话,它是会被重复计算的。
即相同的点集,存在不同的映射方法,而这种映射方法会导致我们的重复计算。
而这种映射方法在匹配的根不同的情况下依旧可能导致相同,如果只根据匹配方的点集是相当麻烦的。
我们不如考虑以某一个点为根的树形态会产生多少的同构,也就是按以这个节点为根的树与我们的原 T 2 T2 T2进行上面 d p dp dp转移的匹配,得到的 h r t h_{rt} hrt表示以 r t rt rt为根的 T 2 T2 T2与原 T 2 T2 T2的同构数。
将原来的 ∑ i = 1 n d p i , r t \sum_{i=1}^{n}dp_{i,rt} i=1ndpi,rt除去 h r t h_{rt} hrt即可,因为原来匹配的就是以 r t rt rt为根的树,它产生的同构树与 h r t h_{rt} hrt相当。
所以这样跑两趟 d p dp dp就可以了。

时间复杂度 O ( ( n + m ) m 2 2 m ) O\left((n+m)m^22^m\right) O((n+m)m22m)

源码

#include<bits/stdc++.h>
using namespace std;
#define MAXN 3005
#define MAXM (1<<10)+5
#define lowbit(x) (x&-x)
#define reg register
#define pb push_back
#define mkpr make_pair
#define fir first
#define sec second
#define debug(x) cerr<<#x<<"="<<x<<'\n'
typedef long long LL;
typedef unsigned long long uLL;     
const LL INF=0x3f3f3f3f3f3f3f3f;  
const int mo=998244353;
const int inv2=499122177;
const int jzm=2333;
const int n1=50;
const int zero=10000;
const int orG=3,invG=332748118;
const double Pi=acos(-1.0);
const double eps=1e-5;
typedef pair<LL,int> pii;
template<typename _T>
_T Fabs(_T x){return x<0?-x:x;}
template<typename _T>
void read(_T &x){_T f=1;x=0;char s=getchar();while(s>'9'||s<'0'){if(s=='-')f=-1;s=getchar();}while('0'<=s&&s<='9'){x=(x<<3)+(x<<1)+(s^48);s=getchar();}x*=f;
}
template<typename _T>
void print(_T x){putchar('\n');while(x>9){putchar((x%10)|'0');x/=10;}putchar(x|'0');}
LL gcd(LL a,LL b){return !b?a:gcd(b,a%b);}
int add(int x,int y,int p){return x+y<p?x+y:x+y-p;}
void Add(int &x,int y,int p){x=add(x,y,p);}
int qkpow(int a,int s,int p){int t=1;while(s){if(s&1LL)t=1ll*a*t%p;a=1ll*a*a%p;s>>=1LL;}return t;}
int n,m,dp[MAXN][12],f[MAXM],tmp[MAXM],g[12][12],ans,ord[15],h[12],sumtr;
vector<int>A[MAXN],B[15],G[15];
void dosaka1(int u,int fa){G[u].clear();int siz=B[u].size();for(int i=0;i<siz;i++){int v=B[u][i];if(v==fa)continue;dosaka1(v,u);G[u].pb(v);}
}
void dosaka2(int u,int fa){int siz=A[u].size();for(int i=0;i<siz;i++)if(A[u][i]!=fa)dosaka2(A[u][i],u);for(int i=1;i<=m;i++){f[0]=1;int sz=G[i].size();for(int j=0;j<sz;j++)ord[j]=G[i][j];if(sz)for(int j=0;j<siz;j++){int v=A[u][j];if(v==fa)continue;for(int S=0;S<(1<<sz);S++)tmp[S]=f[S];for(int k=0;k<sz;k++)for(int S=0;S<(1<<sz);S++)if(!(S&(1<<k)))Add(tmp[S|(1<<k)],1ll*dp[v][ord[k]]*f[S]%mo,mo);for(int S=0;S<(1<<sz);S++)f[S]=tmp[S],tmp[S]=0;}dp[u][i]=f[(1<<sz)-1];for(int S=0;S<(1<<sz);S++)f[S]=0;}
}
void dosaka3(int u,int fa){int siz=B[u].size();for(int i=0;i<siz;i++)if(B[u][i]!=fa)dosaka3(B[u][i],u);for(int i=1;i<=m;i++){f[0]=1;int sz=G[i].size();for(int j=0;j<sz;j++)ord[j]=G[i][j];if(sz)for(int j=0;j<siz;j++){int v=B[u][j];if(v==fa)continue;for(int S=0;S<(1<<sz);S++)tmp[S]=f[S];for(int k=0;k<sz;k++)for(int S=0;S<(1<<sz);S++)if(!(S&(1<<k)))Add(tmp[S|(1<<k)],1ll*g[v][ord[k]]*f[S]%mo,mo);for(int S=0;S<(1<<sz);S++)f[S]=tmp[S],tmp[S]=0;}g[u][i]=f[(1<<sz)-1];for(int S=0;S<(1<<sz);S++)f[S]=0;}
}
signed main(){freopen("count.in","r",stdin);freopen("count.out","w",stdout);read(n);for(int i=1;i<n;i++){int u,v;read(u);read(v);A[u].pb(v);A[v].pb(u);}read(m);for(int i=1;i<m;i++){int u,v;read(u);read(v);B[u].pb(v);B[v].pb(u);}for(int rt=1;rt<=m;rt++){dosaka1(rt,0);for(int j=1;j<=m;j++)dosaka3(j,0),Add(h[j],g[j][rt],mo);	}for(int rt=1;rt<=m;rt++){dosaka1(rt,0);dosaka2(1,0);int summ=0;for(int i=1;i<=n;i++)Add(summ,dp[i][rt],mo);Add(ans,1ll*summ%mo*qkpow(h[rt],mo-2,mo)%mo,mo);}printf("%d\n",ans);return 0;
}

谢谢!!!

这篇关于[海军国际项目办公室]数树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/816951

相关文章

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

如何用Docker运行Django项目

本章教程,介绍如何用Docker创建一个Django,并运行能够访问。 一、拉取镜像 这里我们使用python3.11版本的docker镜像 docker pull python:3.11 二、运行容器 这里我们将容器内部的8080端口,映射到宿主机的80端口上。 docker run -itd --name python311 -p

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定

Vue3项目开发——新闻发布管理系统(六)

文章目录 八、首页设计开发1、页面设计2、登录访问拦截实现3、用户基本信息显示①封装用户基本信息获取接口②用户基本信息存储③用户基本信息调用④用户基本信息动态渲染 4、退出功能实现①注册点击事件②添加退出功能③数据清理 5、代码下载 八、首页设计开发 登录成功后,系统就进入了首页。接下来,也就进行首页的开发了。 1、页面设计 系统页面主要分为三部分,左侧为系统的菜单栏,右侧

SpringBoot项目是如何启动

启动步骤 概念 运行main方法,初始化SpringApplication 从spring.factories读取listener ApplicationContentInitializer运行run方法读取环境变量,配置信息创建SpringApplication上下文预初始化上下文,将启动类作为配置类进行读取调用 refresh 加载 IOC容器,加载所有的自动配置类,创建容器在这个过程

Maven创建项目中的groupId, artifactId, 和 version的意思

文章目录 groupIdartifactIdversionname groupId 定义:groupId 是 Maven 项目坐标的第一个部分,它通常表示项目的组织或公司的域名反转写法。例如,如果你为公司 example.com 开发软件,groupId 可能是 com.example。作用:groupId 被用来组织和分组相关的 Maven artifacts,这样可以避免

2. 下载rknn-toolkit2项目

官网链接: https://github.com/airockchip/rknn-toolkit2 安装好git:[[1. Git的安装]] 下载项目: git clone https://github.com/airockchip/rknn-toolkit2.git 或者直接去github下载压缩文件,解压即可。

9.8javaweb项目总结

1.主界面用户信息显示 登录成功后,将用户信息存储在记录在 localStorage中,然后进入界面之前通过js来渲染主界面 存储用户信息 将用户信息渲染在主界面上,并且头像设置跳转,到个人资料界面 这里数据库中还没有设置相关信息 2.模糊查找 检测输入框是否有变更,有的话调用方法,进行查找 发送检测请求,然后接收的时候设置最多显示四个类似的搜索结果

maven发布项目到私服-snapshot快照库和release发布库的区别和作用及maven常用命令

maven发布项目到私服-snapshot快照库和release发布库的区别和作用及maven常用命令 在日常的工作中由于各种原因,会出现这样一种情况,某些项目并没有打包至mvnrepository。如果采用原始直接打包放到lib目录的方式进行处理,便对项目的管理带来一些不必要的麻烦。例如版本升级后需要重新打包并,替换原有jar包等等一些额外的工作量和麻烦。为了避免这些不必要的麻烦,通常我们

html css jquery选项卡 代码练习小项目

在学习 html 和 css jquery 结合使用的时候 做好是能尝试做一些简单的小功能,来提高自己的 逻辑能力,熟悉代码的编写语法 下面分享一段代码 使用html css jquery选项卡 代码练习 <div class="box"><dl class="tab"><dd class="active">手机</dd><dd>家电</dd><dd>服装</dd><dd>数码</dd><dd