\(\mathcal{Description}\)

  Link.

  指定一棵大小为 \(n\),以 \(1\) 为根的有根树的 \(m\) 对邻接关系与 \(q\) 组 \(\text{LCA}\) 关系,求合法树的个数。

  \(0\le m<n\le13\),\(q\le100\)。

\(\mathcal{Solution}\)

  巧妙的状压 owo。不考虑限制,自然地有状态 \(f(u,S)\) 表示用 \(S\) 中的结点构成以 \(u\) 为根的树的方案数。转移相当于划分出一棵子树,有:

\[f(u,S)=\sum_{v\in T\subseteq(S\setminus u)}f(v,T)f(u,S-T)
\]

  不过这样显然会算重复。考虑任意固定子树 \(T\) 内的某个点,设 \(p\not=u\) 且 \(p\in T\),钦定 \(p\in T\) 就避免了重复,则:

\[f(u,S)=\sum_{v,p\in T\subseteq(S\setminus u)}f(v,T)f(u,S-T)
\]

  注意 \(p\) 在求和过程中是常量


  接下来着手处理限制:

  • 限制 \(\text{LCA}\),设当前状态 \(f(w,S)\),枚举到子集 \(T\):

    • 若 \((u\in T)\land(v\in T)\Leftrightarrow\mathrm T\)(注意最后这个罗马字体的 \(\mathrm T\) 表示逻辑运算为真),\(u\) 和 \(v\) 的 \(\text{LCA}\) 必然在 \(T\) 中,所以必然不是 \(w\),矛盾。
    • 若 \(q,r\) 的 \(\text{LCA}\) 指定为 \(p\),且 \(p\in T\land(q\not\in T\lor q\not\in T)\Leftrightarrow \mathrm T\),即两点的 \(\text{LCA}\) 深于其中至少一个点,显然不满足。
  • 限制邻接点,同样地设当前状态 \(f(w,S)\),枚举到子集 \(T\):
    • 若 \(u,v\not=r\) 邻接,且 \((u\in T)\leftrightarrow(v\in T)\Leftrightarrow\mathrm F\),即有且仅有其中一点属于 \(T\),矛盾。
    • 若 \(u,v\) 均与 \(w\) 邻接,且 \((u\in T)\land(v\in T)\Leftrightarrow\mathrm T\),即 \(w\) 向子树 \(T\) 内的至少两个点连边,矛盾。

  转移的时候判一下这四种情况就行啦。

  复杂度 \(\mathcal O(3^nn(n+m+q))\),不过跑不满。为什么我交一发当场最优解 rank1 呢 www?

\(\mathcal{Code}\)

#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;
}

Solution -「CF 599E」Sandy and Nuts的更多相关文章

  1. Solution -「CF 1342E」Placing Rooks

    \(\mathcal{Description}\)   Link.   在一个 \(n\times n\) 的国际象棋棋盘上摆 \(n\) 个车,求满足: 所有格子都可以被攻击到. 恰好存在 \(k\ ...

  2. Solution -「CF 1622F」Quadratic Set

    \(\mathscr{Description}\)   Link.   求 \(S\subseteq\{1,2,\dots,n\}\),使得 \(\prod_{i\in S}i\) 是完全平方数,并最 ...

  3. Solution -「CF 923F」Public Service

    \(\mathscr{Description}\)   Link.   给定两棵含 \(n\) 个结点的树 \(T_1=(V_1,E_1),T_2=(V_2,E_2)\),求一个双射 \(\varph ...

  4. Solution -「CF 923E」Perpetual Subtraction

    \(\mathcal{Description}\)   Link.   有一个整数 \(x\in[0,n]\),初始时以 \(p_i\) 的概率取值 \(i\).进行 \(m\) 轮变换,每次均匀随机 ...

  5. Solution -「CF 1586F」Defender of Childhood Dreams

    \(\mathcal{Description}\)   Link.   定义有向图 \(G=(V,E)\),\(|V|=n\),\(\lang u,v\rang \in E \Leftrightarr ...

  6. Solution -「CF 1237E」Balanced Binary Search Trees

    \(\mathcal{Description}\)   Link.   定义棵点权为 \(1\sim n\) 的二叉搜索树 \(T\) 是 好树,当且仅当: 除去最深的所有叶子后,\(T\) 是满的: ...

  7. Solution -「CF 623E」Transforming Sequence

    题目 题意简述   link.   有一个 \(n\) 个元素的集合,你需要进行 \(m\) 次操作.每次操作选择集合的一个非空子集,要求该集合不是已选集合的并的子集.求操作的方案数,对 \(10^9 ...

  8. Solution -「CF 1023F」Mobile Phone Network

    \(\mathcal{Description}\)   Link.   有一个 \(n\) 个结点的图,并给定 \(m_1\) 条无向带权黑边,\(m_2\) 条无向无权白边.你需要为每条白边指定边权 ...

  9. Solution -「CF 487E」Tourists

    \(\mathcal{Description}\)   Link.   维护一个 \(n\) 个点 \(m\) 条边的简单无向连通图,点有点权.\(q\) 次操作: 修改单点点权. 询问两点所有可能路 ...

随机推荐

  1. JSR-303规范

    规范链接 CONSTRAINT 详细信息 @Valid 被注释的元素是一个对象,需要检查此对象的所有字段值 @Null 被注释的元素必须为 null @NotNull 被注释的元素必须不为 null ...

  2. .net core 和 WPF 开发升讯威在线客服系统:调用百度翻译接口实现实时自动翻译

    业余时间用 .net core 写了一个在线客服系统.并在博客园写了一个系列的文章,写介绍这个开发过程. 我把这款业余时间写的小系统丢在网上,陆续有人找我要私有化版本,我都给了,毕竟软件业的初衷就是免 ...

  3. MySQL提权之udf提权(获得webshell的情况)

    什么是udf提权? MySQL提供了一个让使用者自行添加新的函数的功能,这种用户自行扩展函数的功能就叫udf. 它的提权原理也非常简单!即是利用了root 高权限,创建带有调用cmd的函数的udf.d ...

  4. STM32 EXTI(外部中断)

    一.EXTI 简介 EXTI(External interrupt/event controller)-外部中断/事件控制器,管理了控制器的 20个中断/事件线.每个中断/事件线都对应有一个边沿检测器 ...

  5. ThinkPad S5立体声混响以及语音识别

    smartaudio里面改成语音识别就可以是立体声混响了.但是微软语音识别在国内依然不好用,微软服务在国内太卡了. (联想总是多此一举,各种乱起八糟的软件,给用户造成困难,以前老机子驱动无线网卡锁在L ...

  6. [源码分析] Facebook如何训练超大模型 --- (3)

    [源码分析] Facebook如何训练超大模型 --- (3) 目录 [源码分析] Facebook如何训练超大模型 --- (3) 0x00 摘要 0x01 ZeRO-Offload 1.1 设计原 ...

  7. 浅析Java中的线程池

    Java中的线程池 几乎所有需要异步或并发执行任务的程序都可以使用线程池,开发过程中合理使用线程池能够带来以下三个好处: 降低资源消耗 提高响应速度 提高线程的可管理性 1. 线程池的实现原理 当我们 ...

  8. StringBuilder类介绍

    1 package cn.itcast.p2.stringbuffer.demo; 2 3 public class StringBuilderDemo { 4 public static void ...

  9. mesos是什么

    听过不少人在讨论 Mesos,然而并不是很明白 Mesos 到底能够解决什么问题,使用场景是怎样的,周伟涛(国内较早一批接触使用 Docker,Mesos 等技术的开发者)用一句话形容它, Mesos ...

  10. keepalived的抢占与非抢占模式

    目录 一:keepalived的抢占与非抢占模式 1.抢占模式 2.非抢占模式 二:接下来分4种情况说明 三:以上3种,只要级别高就会获取master,与state状态是无关的 一:keepalive ...