Solution -「CF 599E」Sandy and Nuts
Link.
指定一棵大小为 ,以 为根的有根树的 对邻接关系与 组 关系,求合法树的个数。
,。
巧妙的状压 owo。不考虑限制,自然地有状态 表示用 中的结点构成以 为根的树的方案数。转移相当于划分出一棵子树,有:
不过这样显然会算重复。考虑任意固定子树 内的某个点,设 且 ,钦定 就避免了重复,则:
注意 在求和过程中是常量。
接下来着手处理限制:
- 限制 ,设当前状态 ,枚举到子集 :
- 若 (注意最后这个罗马字体的 表示逻辑运算为真), 和 的 必然在 中,所以必然不是 ,矛盾。
- 若 的 指定为 ,且 ,即两点的 深于其中至少一个点,显然不满足。
- 限制邻接点,同样地设当前状态 ,枚举到子集 :
- 若 邻接,且 ,即有且仅有其中一点属于 ,矛盾。
- 若 均与 邻接,且 ,即 向子树 内的至少两个点连边,矛盾。
转移的时候判一下这四种情况就行啦。
复杂度 ,不过跑不满。为什么我交一发当场最优解 rank1 呢 www?
#include <cstdio>
#include <vector>
#include <cstring>
#define bel( x, S ) ( ( S >> x ) & 1 )
typedef long long LL;
typedef std::pair<int, int> pii;
const int MAXN = 13;
int n, m, q;
LL f[MAXN + 5][1 << MAXN];
std::vector<int> adj[MAXN + 5];
std::vector<pii> dif[MAXN + 5];
inline bool check ( const int r, const int T ) {
for ( pii p: dif[r] ) { // LCA情况1.
if ( bel ( p.first, T ) && bel ( p.second, T ) ) {
return false;
}
}
for ( int u = 0; u < n; ++ u ) { // LCA情况2.
if ( ! ( ( T >> u ) & 1 ) ) continue;
for ( pii p: dif[u] ) {
if ( ! bel ( p.first, T ) || ! bel ( p.second, T ) ) {
return false;
}
}
}
for ( int u = 0; u < n; ++ u ) { // 邻接情况1.
if ( u == r ) continue;
for ( int v: adj[u] ) {
if ( v ^ r && bel ( u, T ) ^ bel ( v, T ) ) {
return false;
}
}
}
int cnt = 0;
for ( int u: adj[r] ) cnt += bel ( u, T ); // 邻接情况2.
return cnt <= 1;
}
inline LL solve ( const int r, int S ) {
LL& ret = f[r][S];
if ( ~ ret ) return ret;
ret = 0, S ^= 1 << r; // 这里注意去除根节点.
int p;
for ( p = 0; p < n && ! ( ( S >> p ) & 1 ); ++ p ); // 钦定一点p.
for ( int T = S; T; T = ( T - 1 ) & S ) { // 枚举子集.
if ( ! ( ( T >> p ) & 1 ) || ! check ( r, T ) ) continue;
for ( int u = 0; u < n; ++ u ) {
if ( ! bel ( u, T ) ) continue;
ret += solve ( u, T ) * solve ( r, S ^ T ^ ( 1 << r ) );
}
}
return ret;
}
int main () {
scanf ( "%d %d %d", &n, &m, &q );
for ( int i = 1, u, v; i <= m; ++ i ) {
scanf ( "%d %d", &u, &v ), -- u, -- v;
adj[u].push_back ( v ), adj[v].push_back ( u );
}
for ( int i = 1, u, v, w; i <= q; ++ i ) {
scanf ( "%d %d %d", &u, &v, &w ), -- u, -- v, -- w;
dif[w].push_back ( { u, v } );
}
memset ( f, 0xff, sizeof f );
for ( int u = 0; u < n; ++ u ) f[u][1 << u] = 1; // 单点,一种方案.
printf ( "%lld\n", solve ( 0, ( 1 << n ) - 1 ) );
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现